| 1 | rodolico | 1 | <?php
 | 
        
           |  |  | 2 |   | 
        
           | 14 | rodolico | 3 | /*
 | 
        
           |  |  | 4 |    Author: R. W. Rodolico
 | 
        
           |  |  | 5 |    Date:   20090403
 | 
        
           |  |  | 6 |   | 
        
           |  |  | 7 |    Class which holds basic reports for generation in client programs.
 | 
        
           |  |  | 8 |    Assumes a database table of at least the following format (all additional columns ignored)
 | 
        
           |  |  | 9 |       create table report (
 | 
        
           |  |  | 10 |          report_id   int unsigned not null auto_increment,
 | 
        
           |  |  | 11 |          name        varchar(20) not null comment 'Display Name of Report',
 | 
        
           |  |  | 12 |          query       text not null comment 'Query to be executed',
 | 
        
           |  |  | 13 |          parameters  text comment 'All parameters used in above',
 | 
        
           |  |  | 14 |          primary key (report_id)
 | 
        
           |  |  | 15 |       ) comment 'holds definition for report';
 | 
        
           |  |  | 16 |   | 
        
           |  |  | 17 |    Columns are defined as:
 | 
        
           |  |  | 18 |       name -- This is the name of the report for display. This is displayed by Report::listAllReports
 | 
        
           |  |  | 19 |       query -- an SQL Query for the report. Parameters may be inserted by surrounding them with '<' and '>'
 | 
        
           |  |  | 20 |                and these parameters will be replaced by user input values on execution.
 | 
        
           |  |  | 21 |       parameters -- List of parameter information to form query. These are one parameter per line. Each line
 | 
        
           |  |  | 22 |                     is a $parametersDelimiter delimited string of data.
 | 
        
           |  |  | 23 |   | 
        
           |  |  | 24 |                        1 - user prompt for value
 | 
        
           |  |  | 25 |                        2 - query.
 | 
        
           |  |  | 26 |                        3 - displayQuery
 | 
        
           |  |  | 27 |   | 
        
           |  |  | 28 |     This class is free software: you can redistribute it and/or modify
 | 
        
           |  |  | 29 |     it under the terms of the GNU General Public License as published by
 | 
        
           |  |  | 30 |     the Free Software Foundation, either version 3 of the License, or
 | 
        
           |  |  | 31 |     (at your option) any later version.
 | 
        
           |  |  | 32 |   | 
        
           |  |  | 33 |     This program is distributed in the hope that it will be useful,
 | 
        
           |  |  | 34 |     but WITHOUT ANY WARRANTY; without even the implied warranty of
 | 
        
           |  |  | 35 |     MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
 | 
        
           |  |  | 36 |     GNU General Public License for more details.
 | 
        
           |  |  | 37 |   | 
        
           |  |  | 38 |     You should have received a copy of the GNU General Public License
 | 
        
           |  |  | 39 |     along with this program.  If not, see <http://www.gnu.org/licenses/>                       
 | 
        
           |  |  | 40 | */
 | 
        
           |  |  | 41 |   | 
        
           |  |  | 42 |    /* class QueryParameters
 | 
        
           |  |  | 43 |       Designed as a simple helper class to Reports. It simplified the writing of the Reports class by abstracting
 | 
        
           |  |  | 44 |       all of the parameters functions. The parameters are a very complex topic in and of themselves.
 | 
        
           |  |  | 45 |    */
 | 
        
           | 1 | rodolico | 46 |    class QueryParameters {
 | 
        
           | 14 | rodolico | 47 |       protected $parameter;    // parameter name. matches <param_name> in report query
 | 
        
           |  |  | 48 |       protected $prompt;       // user prompt when requesting a value for this parameter
 | 
        
           |  |  | 49 |       protected $query;        // if defined, generates a <select> block to acquire the value for this parameter instead of an <INPUT> block
 | 
        
           |  |  | 50 |       protected $displayQuery; // if defined, result is concated with any other displayQuery results for the title of the report
 | 
        
           |  |  | 51 |       protected $value;        // value used for displayQuery. See documentation
 | 
        
           |  |  | 52 |       private $parametersDelimiter = '++'; // parameter delimiter
 | 
        
           | 1 | rodolico | 53 |   | 
        
           |  |  | 54 |       function __constructor ( $parameter = '', $prompt = '', $query = '' ) {
 | 
        
           |  |  | 55 |          $this->parameter = $parameter;
 | 
        
           |  |  | 56 |          $this->prompt = $prompt;
 | 
        
           |  |  | 57 |          $this->query = $query;
 | 
        
           |  |  | 58 |       }
 | 
        
           |  |  | 59 |   | 
        
           | 14 | rodolico | 60 |       /* following functions are setters/getters for the class members */
 | 
        
           |  |  | 61 |       function parametersDelimiter ( $newDelimiter = '' ) {
 | 
        
           |  |  | 62 |          $oldValue = $this->parametersDelimiter;
 | 
        
           |  |  | 63 |          if ($newDelimiter) {
 | 
        
           |  |  | 64 |             $this->parametersDelimiter = $newDelimiter;
 | 
        
           |  |  | 65 |          }
 | 
        
           |  |  | 66 |          return $oldValue;
 | 
        
           | 1 | rodolico | 67 |       }
 | 
        
           |  |  | 68 |   | 
        
           |  |  | 69 |       function name ( $newParameter = '' ) {
 | 
        
           |  |  | 70 |          $oldParameter = $this->parameter;
 | 
        
           |  |  | 71 |          if ($newParameter) {
 | 
        
           |  |  | 72 |             $this->parameter = $newParameter;
 | 
        
           |  |  | 73 |          }
 | 
        
           |  |  | 74 |          return $oldParameter;
 | 
        
           |  |  | 75 |       }
 | 
        
           |  |  | 76 |   | 
        
           |  |  | 77 |       function value ( $newValue = '' ) {
 | 
        
           |  |  | 78 |          $oldValue = $this->value;
 | 
        
           |  |  | 79 |          if ($newValue) {
 | 
        
           |  |  | 80 |             $this->value = $newValue;
 | 
        
           |  |  | 81 |          }
 | 
        
           |  |  | 82 |          return $oldValue;
 | 
        
           |  |  | 83 |       }
 | 
        
           |  |  | 84 |   | 
        
           |  |  | 85 |       function prompt ( $newPrompt = '' ) {
 | 
        
           |  |  | 86 |          $oldPrompt = $this->prompt;
 | 
        
           |  |  | 87 |          if ($newPrompt) {
 | 
        
           |  |  | 88 |             $this->prompt = $newPrompt;
 | 
        
           |  |  | 89 |          }
 | 
        
           |  |  | 90 |          return $oldPrompt;
 | 
        
           |  |  | 91 |       }
 | 
        
           |  |  | 92 |   | 
        
           |  |  | 93 |       function query ( $newQuery = '' ) {
 | 
        
           |  |  | 94 |          $oldQuery = $this->query;
 | 
        
           |  |  | 95 |          if ($newQuery) {
 | 
        
           |  |  | 96 |             $this->query = $newQuery;
 | 
        
           |  |  | 97 |          }
 | 
        
           |  |  | 98 |          return $oldQuery;
 | 
        
           |  |  | 99 |       }
 | 
        
           |  |  | 100 |   | 
        
           |  |  | 101 |       function displayQuery ( $newQuery = '' ) {
 | 
        
           |  |  | 102 |          $oldQuery = $this->displayQuery;
 | 
        
           |  |  | 103 |          if ($newQuery) {
 | 
        
           |  |  | 104 |             $this->displayQuery = $newQuery;
 | 
        
           |  |  | 105 |          }
 | 
        
           |  |  | 106 |          return $oldQuery;
 | 
        
           |  |  | 107 |       }
 | 
        
           |  |  | 108 |   | 
        
           | 14 | rodolico | 109 |       // takes a single line of a parameters entry and breaks it into its values
 | 
        
           |  |  | 110 |       function stringToParameter( $string ) {
 | 
        
           |  |  | 111 |          $temp = explode( $this->parametersDelimiter, $string );
 | 
        
           |  |  | 112 |          $this->parameter = $temp[0];
 | 
        
           |  |  | 113 |          $this->prompt = $temp[1];
 | 
        
           |  |  | 114 |          $this->query = $temp[2];
 | 
        
           |  |  | 115 |          $this->displayQuery = $temp[3];
 | 
        
           |  |  | 116 |       }
 | 
        
           |  |  | 117 |   | 
        
           |  |  | 118 |       /* 
 | 
        
           |  |  | 119 |          creates a td with the prompt, and an <input> enter the value for that parameter. 
 | 
        
           |  |  | 120 |          If query is defined, will generate a <select> instead
 | 
        
           |  |  | 121 |       */
 | 
        
           | 1 | rodolico | 122 |       function toHTML () {
 | 
        
           |  |  | 123 |          $result = "<td>$this->prompt</td><td>";
 | 
        
           |  |  | 124 |          if ( $this->query ) {
 | 
        
           |  |  | 125 |             $result .= "<select name='report_param_$this->parameter'>" . queryToSelect( $this->query ) . '</select>';
 | 
        
           |  |  | 126 |          } else {
 | 
        
           | 23 | rodolico | 127 |             $result .= "<input type='text' name='report_param_$this->parameter'>";
 | 
        
           | 1 | rodolico | 128 |          }
 | 
        
           |  |  | 129 |          $result .= '</td>';
 | 
        
           |  |  | 130 |          return $result;
 | 
        
           |  |  | 131 |       }
 | 
        
           |  |  | 132 |    } // class QueryParameters
 | 
        
           | 14 | rodolico | 133 |   | 
        
           |  |  | 134 |    /* 
 | 
        
           |  |  | 135 |       the Report class is designed to be semi-automatic. It can, with little code, create a <select> containing the available reports,
 | 
        
           |  |  | 136 |       display any parameters required for that report, then run the report and return a table containing the results.
 | 
        
           |  |  | 137 |    */
 | 
        
           | 1 | rodolico | 138 |   | 
        
           |  |  | 139 |    class Report {
 | 
        
           | 14 | rodolico | 140 |       protected $name;  // name of report
 | 
        
           |  |  | 141 |       protected $title; // title for the report
 | 
        
           |  |  | 142 |       protected $query; // query to generate the report
 | 
        
           |  |  | 143 |       protected $parameters = array(); // array of parameters entries for the report. Class QueryParameters
 | 
        
           | 1 | rodolico | 144 |       private $tableName = 'report';
 | 
        
           |  |  | 145 |       private $nameField = 'name';
 | 
        
           |  |  | 146 |       private $queryField = 'query';
 | 
        
           |  |  | 147 |       private $parametersField = 'parameters';
 | 
        
           |  |  | 148 |       private $idField = 'report_id';
 | 
        
           |  |  | 149 |   | 
        
           | 14 | rodolico | 150 |       /* will simply get a list of all reports from the table and allow the user to select one */
 | 
        
           | 1 | rodolico | 151 |       static public function listAllReports( $tableName = 'report', $nameField = 'name', $idField = 'report_id' ) {
 | 
        
           |  |  | 152 |          return queryToSelect( "select $idField, $nameField from $tableName" );
 | 
        
           |  |  | 153 |       }
 | 
        
           |  |  | 154 |   | 
        
           |  |  | 155 |       function __constructor ($name = '', $title = '', $query = '', $parameters = array() ) {
 | 
        
           |  |  | 156 |          $this->name = $name;
 | 
        
           |  |  | 157 |          $this->title = $title;
 | 
        
           |  |  | 158 |          $this->query = $query;
 | 
        
           |  |  | 159 |          $this->parameters = $parameters;
 | 
        
           |  |  | 160 |       }
 | 
        
           |  |  | 161 |   | 
        
           | 14 | rodolico | 162 |       /* setters/getters for the various class members */
 | 
        
           | 1 | rodolico | 163 |       function name ( $newName = '' ) {
 | 
        
           |  |  | 164 |          $oldName = $this->name;
 | 
        
           |  |  | 165 |          if ($newName) {
 | 
        
           |  |  | 166 |             $this->name = $newName;
 | 
        
           |  |  | 167 |          }
 | 
        
           |  |  | 168 |          return $oldName;
 | 
        
           |  |  | 169 |       }
 | 
        
           |  |  | 170 |   | 
        
           |  |  | 171 |       function query ( $newQuery = '' ) {
 | 
        
           |  |  | 172 |          $oldQuery = $this->query;
 | 
        
           |  |  | 173 |          if ($newQuery) {
 | 
        
           |  |  | 174 |             $this->query = $newQuery;
 | 
        
           |  |  | 175 |          }
 | 
        
           |  |  | 176 |          return $oldQuery;
 | 
        
           |  |  | 177 |       }
 | 
        
           |  |  | 178 |   | 
        
           |  |  | 179 |       function parameters ( $newParameters = array() ) {
 | 
        
           |  |  | 180 |          $oldParameters = $this->parameters;
 | 
        
           |  |  | 181 |          if ($newParameters) {
 | 
        
           |  |  | 182 |             $this->parameters = $newParameters;
 | 
        
           |  |  | 183 |          }
 | 
        
           |  |  | 184 |          return $oldParameters;
 | 
        
           |  |  | 185 |       }
 | 
        
           |  |  | 186 |   | 
        
           |  |  | 187 |       function addParameter ( $newParameter = array() ) {
 | 
        
           |  |  | 188 |          $this->parameters[] = $newParameter;
 | 
        
           |  |  | 189 |       }
 | 
        
           |  |  | 190 |   | 
        
           | 14 | rodolico | 191 |       /* get the report definition from the database. Will decipher the definition, filling out all class members with their values */
 | 
        
           |  |  | 192 |       function loadFromDatabase ( $id ) {
 | 
        
           |  |  | 193 |          $sql = "select $this->nameField, $this->queryField, $this->parametersField from $this->tableName where $this->idField = $id";
 | 
        
           |  |  | 194 |          $result = queryDatabaseExtended( $sql );
 | 
        
           |  |  | 195 |          $result = $result['data'][0];
 | 
        
           |  |  | 196 |          $this->name = $result[$this->nameField];
 | 
        
           |  |  | 197 |          $this->title = $result[$this->nameField];
 | 
        
           |  |  | 198 |          $this->query = $result[$this->queryField];
 | 
        
           |  |  | 199 |          $parameters = $result[$this->parametersField];
 | 
        
           |  |  | 200 |          $parameters = explode("\n", $parameters);
 | 
        
           |  |  | 201 |          foreach ( $parameters as $thisParam ) {
 | 
        
           |  |  | 202 |             if (! $thisParam ) {
 | 
        
           |  |  | 203 |                continue;
 | 
        
           |  |  | 204 |             }
 | 
        
           |  |  | 205 |             $temp = new QueryParameters();
 | 
        
           |  |  | 206 |             $temp->stringToParameter($thisParam);
 | 
        
           |  |  | 207 |             $this->parameters[] = $temp;
 | 
        
           |  |  | 208 |          }
 | 
        
           |  |  | 209 |       }
 | 
        
           |  |  | 210 |   | 
        
           |  |  | 211 |       /* 
 | 
        
           |  |  | 212 |          runs the report
 | 
        
           | 20 | rodolico | 213 |          $parameters, if passed in, is an array of paramtername=>value and will be used in place of any values on the form
 | 
        
           |  |  | 214 |          $additionalLimitations is a partial where clause. When passed in, it will replace the special parameter <additionalLimitations>.
 | 
        
           |  |  | 215 |                          if it is not passed in, <additionalLimitations> will be removed
 | 
        
           | 25 | rodolico | 216 |          $makeTitle, if set to false, will simply return the data in a table. If true, will return the title and the data
 | 
        
           | 14 | rodolico | 217 |          The return value is the title in an H1, the conditions in H2's, and the results in a table
 | 
        
           |  |  | 218 |        */
 | 
        
           | 25 | rodolico | 219 |       function run( $parameters = '', $additionalLimitations = '', $makeTitle = true ) {
 | 
        
           | 1 | rodolico | 220 |          $conditions = array();
 | 
        
           | 26 | rodolico | 221 |          foreach ( $this->parameters as $parameter ) {
 | 
        
           |  |  | 222 |             if ( $parameters[$parameter->name()] ) { // in the array passed in
 | 
        
           |  |  | 223 |                $parameter->value( $parameters[$parameter->name()] );
 | 
        
           |  |  | 224 |             } elseif ($_POST['report_param_' . $parameter->name()]) { // in $_POST
 | 
        
           |  |  | 225 |                $parameter->value($_POST['report_param_' . $parameter->name()]);
 | 
        
           | 1 | rodolico | 226 |             }
 | 
        
           |  |  | 227 |          }
 | 
        
           | 23 | rodolico | 228 |          // print "<pre>"; print_r( $this ); print "</pre>";
 | 
        
           | 1 | rodolico | 229 |          foreach ( $this->parameters as $parameter ) {
 | 
        
           |  |  | 230 |             $toFind = '<' . $parameter->name() . '>';
 | 
        
           |  |  | 231 |             $this->query = preg_replace( "/$toFind/", $parameter->value(), $this->query );
 | 
        
           |  |  | 232 |             if ( $parameter->displayQuery() ) {
 | 
        
           |  |  | 233 |                //$result = preg_replace( '/<value>/', $parameter->value(), $parameter->displayQuery() );
 | 
        
           | 23 | rodolico | 234 |                $sql = preg_replace( '/<value>/', $parameter->value(), $parameter->displayQuery() );
 | 
        
           |  |  | 235 |                // print "<pre>$sql</pre>\n";
 | 
        
           |  |  | 236 |                $result = queryDatabaseExtended($sql , false);
 | 
        
           | 1 | rodolico | 237 |                //print "<pre>" . print_r( $result ); print "</pre>";
 | 
        
           |  |  | 238 |                $conditions[] = $parameter->name() . ' - '  . $result['data'][0][0];
 | 
        
           |  |  | 239 |             } else {
 | 
        
           |  |  | 240 |                $conditions[] = $parameter->name() . ' - ' . $parameter->value();
 | 
        
           |  |  | 241 |             }
 | 
        
           |  |  | 242 |          }
 | 
        
           | 20 | rodolico | 243 |          if ($additionalLimitations) { // the user wants us to limit additionally
 | 
        
           | 21 | rodolico | 244 |             $this->query = preg_replace( '/<additionalLimitations>/', ' and ' . $additionalLimitations, $this->query );
 | 
        
           | 20 | rodolico | 245 |          } else {
 | 
        
           | 21 | rodolico | 246 |             $this->query = preg_replace( '/<additionalLimitations>/', '', $this->query );
 | 
        
           | 20 | rodolico | 247 |          }
 | 
        
           | 23 | rodolico | 248 |          // print "<pre>$this->query</pre>";
 | 
        
           | 25 | rodolico | 249 |          return ( $makeTitle ? '<h1>' . $this->title . '</h1>' . '<h2>' . implode('</h2><h2>', $conditions) . '</h2>' : '' ) . queryToTable($this->query);
 | 
        
           | 1 | rodolico | 250 |       }
 | 
        
           |  |  | 251 |   | 
        
           | 14 | rodolico | 252 |       /*
 | 
        
           |  |  | 253 |          Creates a table with the necessary INPUT or SELECT fields to get the parameter values from the user
 | 
        
           |  |  | 254 |          returns a table suitable for inserting into a FORM
 | 
        
           |  |  | 255 |       */
 | 
        
           |  |  | 256 |       function parameterInputScreen () {
 | 
        
           | 1 | rodolico | 257 |          $result = "<table border='1'>\n";
 | 
        
           |  |  | 258 |          $result .= "<tr><td colspan='2'>$this->name</td></tr>";
 | 
        
           |  |  | 259 |          foreach ( $this->parameters as $parameter ) {
 | 
        
           |  |  | 260 |             $result .= '<tr>' . $parameter->toHTML() . '</tr>';
 | 
        
           |  |  | 261 |          }
 | 
        
           | 23 | rodolico | 262 |          return $result . '</table>';
 | 
        
           | 1 | rodolico | 263 |       }
 | 
        
           | 14 | rodolico | 264 |   | 
        
           |  |  | 265 |       // kept because some code used the old name. I messed up when I wrote the original. I don't think the name is indicative of the
 | 
        
           |  |  | 266 |       // use of this function. This will be removed in the future.
 | 
        
           |  |  | 267 |       function toHTML( ) {
 | 
        
           |  |  | 268 |          return parameterInputScreen();
 | 
        
           |  |  | 269 |       }
 | 
        
           | 1 | rodolico | 270 |   | 
        
           |  |  | 271 |    } // class Report
 | 
        
           |  |  | 272 |   | 
        
           |  |  | 273 | ?>
 |