FrxDataSource.inc in Forena Reports 7.4
Same filename and directory in other branches
Class that defines default methods for access control in an FrxDataSource
File
FrxDataSource.incView source
<?php
// $Id$
require_once 'forena.common.inc';
/**
* @file
* Class that defines default methods for access control in an FrxDataSource
*
*/
class FrxDataSource {
public $name;
public $conf;
public $block_path;
public $comment_prefix;
public $comment_suffix;
public $block_ext;
public $block_extensions;
public $types;
public $block_name;
public $db_type = 'mysql';
protected $te;
public $debug = FALSE;
public function __construct($conf, $repos_path, $name) {
$this->conf = $conf;
$this->comment_prefix = '--';
$this->block_ext = 'sql';
$this->block_path = $repos_path;
$this->app = FrxReportGenerator::instance()->app;
$this->block_extensions = array(
'inc',
'sql',
'xml',
);
$this->name = $name;
$this->debug = @$conf['debug'];
}
/**
* Implements the basic default security check of calling
* an access method.
*
* @param unknown_type $arg
* @return unknown
*/
public function access($arg) {
$obj_access = TRUE;
$access = user_access('access ' . $this->name . ' data');
$f = @$this->conf['access callback'];
if ($arg) {
if ($f && is_callable($f)) {
$obj_access = $f($arg);
}
elseif (isset($this->conf['access block'])) {
$block = @$this->conf['access block'];
$path = '';
if (isset($this->conf['access path'])) {
$path = $this->conf['access path'];
}
$obj_access = Frx::RepoMan()
->blockAccess($block, $path, $arg);
}
}
return $access && $obj_access;
}
protected function loadBlockFromFile($block_name) {
$full_name = $this->name . '/' . $block_name;
$php_class = $block_name;
$php_file = $this->block_path . '/' . $php_class . '.php';
if (Frx::DataFile()
->exists($full_name . '.sql')) {
$contents = Frx::DataFile()
->contents($full_name . '.sql');
$block = $this
->parseSQLFile($contents);
$block['type'] = 'sql';
}
elseif (Frx::DataFile()
->exists($full_name . '.xml')) {
$contents = Frx::DataFile()
->contents($full_name . '.xml');
$block = $this
->parseXMLFile($contents);
$block['type'] = 'xml';
}
elseif (Frx::DataFile()
->exists($full_name . '.php')) {
include_once $php_file;
if (class_exists($php_class)) {
$o = new $php_class();
$block['type'] = 'php';
$block['access'] = @$o->access;
$block['object'] = $o;
if (method_exists($o, 'tokens')) {
$block['tokens'] = $o
->tokens();
}
elseif (isset($o->tokens)) {
$block['tokens'] = $o->tokens;
}
else {
$block['tokens'] = array();
}
}
}
else {
return array();
}
$block['locked'] = 1;
return $block;
}
/**
* Load blcok data from filesystem
* @param $block_name
*/
function loadBlock($block_name, $include = FALSE) {
if ($include) {
$this->block_name = $block_name;
}
$block = $this
->loadBlockFromFile($block_name);
return $block;
}
/**
* Load tokens from block source
*/
public function tokens($source) {
$tokens = array();
// If we have a regular expression token parser, then get the tokens out of the block.
if ($this->te) {
$tokens = @$this->te
->tokens($source);
$tokens = array_diff($tokens, array(
'current_user',
));
//check tokens in the where clause
}
return $tokens;
}
public function listDBBlocks($search, &$data_blocks) {
$search = '%' . $search . '%';
$sql = 'SELECT * from {forena_data_blocks} WHERE repository=:repos
AND block_name like :search ';
$rs = db_query($sql, array(
':repos' => $this->name,
':search' => $search,
));
foreach ($rs as $block) {
$data_blocks[] = $block->block_name;
}
}
/**
* Find all the blocks matching a provided search string
*
* @param string $search part block names to search for
* @return unknown
*/
public function list_blocks($search, &$block_list, $subdir = '') {
$count = 0;
// First find files that match the search string
$path = $this->block_path . '/';
if ($subdir) {
$path .= $subdir . '/';
}
$block_path = $path . '*' . $search . '*';
// Find sql files
$d = glob($block_path);
if ($d) {
foreach ($d as $file_name) {
// Split off the extention
$p = strripos($file_name, '.');
if ($p !== FALSE) {
$ext = substr($file_name, $p + 1);
$block_name = substr($file_name, 0, $p);
}
else {
$ext = '';
$block_name = $file_name;
}
switch ($ext) {
case 'inc':
require_once $file_name;
$class = str_replace($this->block_path . '/', '', $block_name);
$methods = get_class_methods($class);
if ($methods) {
foreach ($methods as $method) {
if ($method != 'tokens') {
$block_list[] = $class . '.' . $method;
}
}
}
break;
default:
if (array_search($ext, $this->block_extensions) !== FALSE) {
$block_list[] = str_replace($this->block_path . '/', '', $block_name);
}
}
}
}
$count++;
// Find directories
$d = glob($path . '*');
if ($d) {
foreach ($d as $dir_name) {
if (is_dir($dir_name)) {
$dir_name = str_replace($this->block_path . '/', '', $dir_name);
$this
->list_blocks($search, $block_list, $dir_name);
}
}
}
// Date
if (!$subdir && module_exists('forena_query')) {
$this
->listDBBlocks($search, $block_list);
}
}
public function debug($msg = '', $log = '') {
Frx::debug($msg, $log);
}
public function error($msg = '', $log = '') {
if ($msg) {
drupal_set_message($msg, 'error', FALSE);
}
if ($log) {
watchdog('forena', $log, NULL, WATCHDOG_ERROR);
}
}
public function parseXMLFile($contents) {
$comment = $this->comment_prefix;
$trim = '->';
$lines = explode("\n", $contents);
$cnt = count($lines);
$access = '';
$i = 0;
$block = '';
$data = '';
while ($i < $cnt) {
$l = trim($lines[$i], "\r");
@(list($d, $c) = explode($comment, $l, 2));
if ($trim) {
$c = trim($c, $trim);
}
if ($c) {
list($a, $o) = explode('=', $c, 2);
$a = trim($a);
if ($a && $o) {
switch ($a) {
case 'ACCESS':
$access = trim($o);
break;
default:
}
}
}
if (strpos($l, $comment) !== 0) {
$data .= "{$l}\n";
}
$i++;
}
return array(
'access' => $access,
'source' => $data,
'tokens' => $this
->tokens($data),
);
}
public function getSQLInclude($block_name) {
//@TODO: allow relative block includes
$block = $this
->loadBlock($block_name, TRUE);
if ($block && $block['type'] == 'sql') {
return $block;
}
else {
drupal_set_message(t('Include %b.sql not found', array(
'%b' => $block_name,
)), 'error', FALSE);
}
}
public function parseSQLFile($contents) {
$comment = $this->comment_prefix;
$trim = $this->comment_suffix;
$lines = explode("\n", $contents);
$cnt = count($lines);
$access = '';
$i = 0;
$block = '';
$parms = Frx::Data()
->currentContext();
$data = '';
$source = '';
$file = '';
$skip = FALSE;
$in_info = FALSE;
$found_case = FALSE;
$info_text = '';
$tokens = array();
$options = array();
$switch = '';
while ($i < $cnt) {
$l = trim($lines[$i], "\r");
@(list($d, $c) = explode($comment, $l, 2));
if ($trim) {
$c = trim($c, $trim);
}
if ($c) {
$c = trim($c);
@(list($a, $o) = explode('=', $c, 2));
$a = trim($a);
if ($a && $o || $c == 'END' || $c == 'ELSE' || $c == 'INFO') {
if ($c != 'INFO') {
$in_info = false;
}
switch ($a) {
case 'ACCESS':
$access = trim($o);
break;
case 'SWITCH':
$switch = trim($o);
$found_case = FALSE;
break;
case 'CASE':
$match = $this->te
->replace($switch, TRUE) == $this->te
->replace($o);
if ($match) {
$found_case = TRUE;
}
$skip = !$match;
break;
case 'IF':
$skip = !$this->te
->test(trim($o));
break;
case 'END':
$skip = FALSE;
$switch = '';
break;
case 'ELSE':
$skip = $switch ? $found_case : !$skip;
break;
case 'INFO':
$in_info = TRUE;
break;
case 'INCLUDE':
if (!$skip) {
$inc_block = $this
->getSQLInclude(trim($o));
if ($inc_block) {
$data .= $inc_block['source'];
$tokens = array_merge($tokens, $inc_block['tokens']);
}
}
break;
}
}
if ($a != 'ACCESS') {
$file .= "{$l}\n";
}
}
else {
$file .= "{$l}\n";
}
if ($in_info) {
if (strpos($l, $comment) !== 0 && $l) {
$info_text .= "{$l}\n";
}
}
elseif (!$skip) {
if (strpos($l, $comment) !== 0 && $l) {
$data .= "{$l}\n";
}
}
$i++;
}
$tokens = array_merge($tokens, $this
->tokens($contents));
if ($info_text) {
$options = drupal_parse_info_format($info_text);
}
$block = array(
'source' => $data,
'file' => trim($file, " \n"),
'tokens' => $tokens,
'options' => $options,
'access' => $access,
);
return $block;
}
/**
* Dummy method for returning sql data
* @param $parameters array Block parameters
* @param $block array extra block options for data provider.
*/
public function sqlData($parameters, $options = array()) {
return '';
}
/**
* Implement static XML functioin
* @param $source XML Source data from block load
* @param $parm_data Parameter data
*/
public function xmlData($source, $parm_data = '') {
$xml = '';
$xmlData = '';
try {
$xmlData = $source;
$xml = new SimpleXMLElement($xmlData);
} catch (Exception $e) {
$this
->error("Error processing xml\n", $e
->getMessage() . "\n" . $xmlData);
}
return $xml;
}
public function phpData($o, $parameters) {
$data = NULL;
if (is_object($o) && is_callable(array(
$o,
'data',
))) {
$data = $o
->data($parameters);
}
return $data;
}
/**
* Build the SQL clause based on builder data.
* @param $data
*/
public function buildFilterSQL($data) {
$clause = '';
$op = $data['op'];
$i = 0;
if ($data['filter']) {
foreach ($data['filter'] as $cond) {
$i++;
$conj = $i == 1 ? '' : $op . ' ';
if (isset($cond['filter'])) {
$clause .= $conj . ' (' . $this
->buildFilterSQL($cond) . " )\n";
}
else {
switch ($cond['op']) {
case 'IS NULL':
case 'IS NOT NULL':
$expr = $cond['field'] . ' ' . $cond['op'];
break;
default:
$expr = $cond['field'] . ' ' . $cond['op'] . ' ' . $this
->format($cond['value'], $cond['field'], array());
}
$clause .= ' ' . $conj . $expr;
}
}
}
return $clause;
}
/**
* Perform generic type conversion based on attributes.
* @param $key The token key for parameter replacement.
* @param $value The value of the parameter
* @return Ambigous <NULL, array>
*/
public function parmConvert($key, $value) {
if (isset($this->types[$key]) && $this->types[$key]) {
if ($value === NULL || $value === '') {
$value = NULL;
}
else {
switch (strtolower($this->types[$key])) {
case 'date':
$time = @new DateTime($value);
if ($time) {
$value = date_format($time, 'Y-m-d H:i:s');
}
else {
$value = NULL;
}
break;
case 'unixtime':
$time = @new DateTime($value);
if ($time) {
$value = $time
->getTimeStamp();
}
else {
$value = NULL;
}
break;
case 'numeric':
case 'float':
$value = (double) $value;
break;
case 'int':
case 'integer':
$value = (int) $value;
break;
case 'array':
$value = (array) $value;
break;
}
}
}
return $value;
}
/**
* Method to return an array of tables that start with the string
* indicated in $str
* @param $str Name of tables.
* @return multitype:
*/
public function searchTables($str) {
return array();
}
public function searchTableColumns($table, $str) {
return array();
}
public function searchTablesSQL() {
switch ($this->db_type) {
case 'mysql':
$sql = "SHOW TABLES LIKE :str";
break;
case 'postgres':
case 'postgresql':
case 'pgsql':
$sql = "SELECT tablename from (\n SELECT schemaname, tablename FROM pg_catalog.pg_tables\n UNION SELECT schemaname, viewname from pg_catalog.pg_views) v\n where schemaname NOT IN ('pg_catalog', 'information_schema') and tablename like :str\n order by 1";
break;
case 'oracle':
case 'oci':
$sql = "SELECT object_name FROM all_objects where object_type in ('TABLE','VIEW')\n AND owner not in ('SYS') AND object_name LIKE :str";
break;
case 'mssql':
$sql = "SELECT table_name FROM INFORMATION_SCHEMA.TABLES WHERE TABLE_TYPE = 'BASE TABLE'\n and table_name like :str";
break;
case 'sqlite':
$sql = 'SELECT name FROM sqlite_master WHERE name like :str';
break;
default:
drupal_set_message(t('Unknown database type: %s', array(
'%s' => $this->db_type,
)), 'error');
}
return $sql;
}
public function searchTableColumnsSQL() {
switch ($this->db_type) {
case 'mysql':
$sql = "select column_name from information_schema.COLUMNS where\n table_schema = :database\n AND table_name = :table AND column_name like :str";
break;
case 'postgres':
case 'postgresql':
case 'pgsql':
$sql = "SELECT column_name from\n information_schema.columns\n WHERE\n table_catalog = :database\n AND table_name = :table\n AND column_name like :str\n order by 1";
break;
case 'oracle':
case 'oci':
$sql = "SELECT column_name FROM all_tab_columns where\n table_name = :table_name\n AND column_name LIKE :str";
break;
case 'mssql':
$sql = "SELECT * FROM INFORMATION_SCHEMA.COLUMNS\n WHERE TABLE_NAME = :table and column_name like :str";
break;
case 'sqlite':
$sql = 'PRAGMA table_info(:table)';
break;
default:
drupal_set_message(t('Unknown database type: %s', array(
'%s' => $this->db_type,
)), 'error');
}
return $sql;
}
}
Classes
Name | Description |
---|---|
FrxDataSource | @file Class that defines default methods for access control in an FrxDataSource |