pictcode / lib / Cake / Model / AclNode.php @ 3bf0e002
履歴 | 表示 | アノテート | ダウンロード (5.505 KB)
| 1 |
<?php
|
|---|---|
| 2 |
/**
|
| 3 |
* CakePHP(tm) : Rapid Development Framework (http://cakephp.org)
|
| 4 |
* Copyright (c) Cake Software Foundation, Inc. (http://cakefoundation.org)
|
| 5 |
*
|
| 6 |
* Licensed under The MIT License
|
| 7 |
* For full copyright and license information, please see the LICENSE.txt
|
| 8 |
* Redistributions of files must retain the above copyright notice.
|
| 9 |
*
|
| 10 |
* @copyright Copyright (c) Cake Software Foundation, Inc. (http://cakefoundation.org)
|
| 11 |
* @link http://cakephp.org CakePHP(tm) Project
|
| 12 |
* @package Cake.Model
|
| 13 |
* @since CakePHP(tm) v 0.2.9
|
| 14 |
* @license http://www.opensource.org/licenses/mit-license.php MIT License
|
| 15 |
*/
|
| 16 |
|
| 17 |
App::uses('Model', 'Model'); |
| 18 |
|
| 19 |
/**
|
| 20 |
* ACL Node
|
| 21 |
*
|
| 22 |
* @package Cake.Model
|
| 23 |
*/
|
| 24 |
class AclNode extends Model { |
| 25 |
|
| 26 |
/**
|
| 27 |
* Explicitly disable in-memory query caching for ACL models
|
| 28 |
*
|
| 29 |
* @var bool
|
| 30 |
*/
|
| 31 |
public $cacheQueries = false; |
| 32 |
|
| 33 |
/**
|
| 34 |
* ACL models use the Tree behavior
|
| 35 |
*
|
| 36 |
* @var array
|
| 37 |
*/
|
| 38 |
public $actsAs = array('Tree' => array('type' => 'nested')); |
| 39 |
|
| 40 |
/**
|
| 41 |
* Constructor
|
| 42 |
*/
|
| 43 |
public function __construct() { |
| 44 |
$config = Configure::read('Acl.database'); |
| 45 |
if (isset($config)) { |
| 46 |
$this->useDbConfig = $config; |
| 47 |
} |
| 48 |
parent::__construct();
|
| 49 |
} |
| 50 |
|
| 51 |
/**
|
| 52 |
* Retrieves the Aro/Aco node for this model
|
| 53 |
*
|
| 54 |
* @param string|array|Model $ref Array with 'model' and 'foreign_key', model object, or string value
|
| 55 |
* @return array Node found in database
|
| 56 |
* @throws CakeException when binding to a model that doesn't exist.
|
| 57 |
*/
|
| 58 |
public function node($ref = null) { |
| 59 |
$db = $this->getDataSource(); |
| 60 |
$type = $this->alias; |
| 61 |
$result = null; |
| 62 |
|
| 63 |
if (!empty($this->useTable)) { |
| 64 |
$table = $this->useTable; |
| 65 |
} else {
|
| 66 |
$table = Inflector::pluralize(Inflector::underscore($type)); |
| 67 |
} |
| 68 |
|
| 69 |
if (empty($ref)) { |
| 70 |
return null; |
| 71 |
} elseif (is_string($ref)) { |
| 72 |
$path = explode('/', $ref); |
| 73 |
$start = $path[0]; |
| 74 |
unset($path[0]); |
| 75 |
|
| 76 |
$queryData = array( |
| 77 |
'conditions' => array( |
| 78 |
$db->name("{$type}.lft") . ' <= ' . $db->name("{$type}0.lft"), |
| 79 |
$db->name("{$type}.rght") . ' >= ' . $db->name("{$type}0.rght")), |
| 80 |
'fields' => array('id', 'parent_id', 'model', 'foreign_key', 'alias'), |
| 81 |
'joins' => array(array( |
| 82 |
'table' => $table, |
| 83 |
'alias' => "{$type}0", |
| 84 |
'type' => 'INNER', |
| 85 |
'conditions' => array("{$type}0.alias" => $start) |
| 86 |
)), |
| 87 |
'order' => $db->name("{$type}.lft") . ' DESC' |
| 88 |
); |
| 89 |
|
| 90 |
$conditionsAfterJoin = array(); |
| 91 |
|
| 92 |
foreach ($path as $i => $alias) { |
| 93 |
$j = $i - 1; |
| 94 |
|
| 95 |
$queryData['joins'][] = array( |
| 96 |
'table' => $table, |
| 97 |
'alias' => "{$type}{$i}", |
| 98 |
'type' => 'INNER', |
| 99 |
'conditions' => array( |
| 100 |
$db->name("{$type}{$i}.alias") . ' = ' . $db->value($alias, 'string') |
| 101 |
) |
| 102 |
); |
| 103 |
|
| 104 |
// it will be better if this conditions will performs after join operation
|
| 105 |
$conditionsAfterJoin[] = $db->name("{$type}{$j}.id") . ' = ' . $db->name("{$type}{$i}.parent_id"); |
| 106 |
$conditionsAfterJoin[] = $db->name("{$type}{$i}.rght") . ' < ' . $db->name("{$type}{$j}.rght"); |
| 107 |
$conditionsAfterJoin[] = $db->name("{$type}{$i}.lft") . ' > ' . $db->name("{$type}{$j}.lft"); |
| 108 |
|
| 109 |
$queryData['conditions'] = array('or' => array( |
| 110 |
$db->name("{$type}.lft") . ' <= ' . $db->name("{$type}0.lft") . ' AND ' . $db->name("{$type}.rght") . ' >= ' . $db->name("{$type}0.rght"), |
| 111 |
$db->name("{$type}.lft") . ' <= ' . $db->name("{$type}{$i}.lft") . ' AND ' . $db->name("{$type}.rght") . ' >= ' . $db->name("{$type}{$i}.rght")) |
| 112 |
); |
| 113 |
} |
| 114 |
$queryData['conditions'] = array_merge($queryData['conditions'], $conditionsAfterJoin); |
| 115 |
$result = $db->read($this, $queryData, -1); |
| 116 |
$path = array_values($path); |
| 117 |
|
| 118 |
if (!isset($result[0][$type]) || |
| 119 |
(!empty($path) && $result[0][$type]['alias'] != $path[count($path) - 1]) || |
| 120 |
(empty($path) && $result[0][$type]['alias'] != $start) |
| 121 |
) {
|
| 122 |
return false; |
| 123 |
} |
| 124 |
} elseif (is_object($ref) && $ref instanceof Model) { |
| 125 |
$ref = array('model' => $ref->name, 'foreign_key' => $ref->id); |
| 126 |
} elseif (is_array($ref) && !(isset($ref['model']) && isset($ref['foreign_key']))) { |
| 127 |
$name = key($ref); |
| 128 |
list(, $alias) = pluginSplit($name); |
| 129 |
|
| 130 |
$model = ClassRegistry::init(array('class' => $name, 'alias' => $alias)); |
| 131 |
|
| 132 |
if (empty($model)) { |
| 133 |
throw new CakeException('cake_dev', "Model class '%s' not found in AclNode::node() when trying to bind %s object", $type, $this->alias); |
| 134 |
} |
| 135 |
|
| 136 |
$tmpRef = null; |
| 137 |
if (method_exists($model, 'bindNode')) { |
| 138 |
$tmpRef = $model->bindNode($ref); |
| 139 |
} |
| 140 |
if (empty($tmpRef)) { |
| 141 |
$ref = array('model' => $alias, 'foreign_key' => $ref[$name][$model->primaryKey]); |
| 142 |
} else {
|
| 143 |
if (is_string($tmpRef)) { |
| 144 |
return $this->node($tmpRef); |
| 145 |
} |
| 146 |
$ref = $tmpRef; |
| 147 |
} |
| 148 |
} |
| 149 |
if (is_array($ref)) { |
| 150 |
if (is_array(current($ref)) && is_string(key($ref))) { |
| 151 |
$name = key($ref); |
| 152 |
$ref = current($ref); |
| 153 |
} |
| 154 |
foreach ($ref as $key => $val) { |
| 155 |
if (strpos($key, $type) !== 0 && strpos($key, '.') === false) { |
| 156 |
unset($ref[$key]); |
| 157 |
$ref["{$type}0.{$key}"] = $val; |
| 158 |
} |
| 159 |
} |
| 160 |
$queryData = array( |
| 161 |
'conditions' => $ref, |
| 162 |
'fields' => array('id', 'parent_id', 'model', 'foreign_key', 'alias'), |
| 163 |
'joins' => array(array( |
| 164 |
'table' => $table, |
| 165 |
'alias' => "{$type}0", |
| 166 |
'type' => 'INNER', |
| 167 |
'conditions' => array( |
| 168 |
$db->name("{$type}.lft") . ' <= ' . $db->name("{$type}0.lft"), |
| 169 |
$db->name("{$type}.rght") . ' >= ' . $db->name("{$type}0.rght") |
| 170 |
) |
| 171 |
)), |
| 172 |
'order' => $db->name("{$type}.lft") . ' DESC' |
| 173 |
); |
| 174 |
$result = $db->read($this, $queryData, -1); |
| 175 |
|
| 176 |
if (!$result) { |
| 177 |
throw new CakeException(__d('cake_dev', "AclNode::node() - Couldn't find %s node identified by \"%s\"", $type, print_r($ref, true))); |
| 178 |
} |
| 179 |
} |
| 180 |
return $result; |
| 181 |
} |
| 182 |
|
| 183 |
} |