2]` Matches elements that have an `id` index greater than 2. * * There are a number of attribute operators: * * - `=`, `!=` Equality. * - `>`, `<`, `>=`, `<=` Value comparison. * - `=/.../` Regular expression pattern match. * * Given a set of User array data, from a `$User->find('all')` call: * * - `1.User.name` Get the name of the user at index 1. * - `{n}.User.name` Get the name of every user in the set of users. * - `{n}.User[id]` Get the name of every user with an id key. * - `{n}.User[id>=2]` Get the name of every user with an id key greater than or equal to 2. * - `{n}.User[username=/^paul/]` Get User elements with username containing `^paul`. * * @param array $data The data to extract from. * @param string $path The path to extract. * @return array An array of the extracted values. Returns an empty array * if there are no matches. */ public static function extract(array $data, $path) { if (empty($path)) { return $data; } // Simple paths. if (!preg_match('/[{\[]/', $path)) { return (array) self::get($data, $path); } if (strpos('[', $path) === false) { $tokens = explode('.', $path); } else { $tokens = String::tokenize($path, '.', '[', ']'); } $_key = '__set_item__'; $context = array($_key => array($data)); do { $token = array_shift($tokens); $next = array(); $conditions = false; $position = strpos($token, '['); if ($position !== false) { $conditions = substr($token, $position); $token = substr($token, 0, $position); } foreach ($context[$_key] as $item) { foreach ($item as $k => $v) { if (self::_matchToken($k, $token)) { $next[] = $v; } } } // Filter for attributes. if ($conditions) { $filter = array(); foreach ($next as $item) { if (self::_matches($item, $conditions)) { $filter[] = $item; } } $next = $filter; } $context = array($_key => $next); } while (!empty($tokens)); return $context[$_key]; } /** * Check a key against a token. * * @param string $key The key in the array being searched. * @param string $token The token being matched. * @return boolean */ protected static function _matchToken($key, $token) { if ($token === '{n}') { return is_numeric($key); } elseif ($token === '{s}') { return is_string($key); } elseif (is_numeric($token)) { return ($key == $token); } else { return ($key === $token); } } /** * Checks whether or not $data matches the selector * * @param array $data Array of data to match. * @param string $selector The selector to match. * @return boolean Fitness of expression. */ protected static function _matches(array $data, $selector) { preg_match_all( '/(\[ (?[^=>[><]) \s* (?[^\]]+) )? \])/x', $selector, $conditions, PREG_SET_ORDER ); $ok = true; while ($ok) { if (empty($conditions)) { break; } $cond = array_shift($conditions); $attr = $cond['attr']; $op = isset($cond['op']) ? $cond['op'] : null; $val = isset($cond['val']) ? $cond['val'] : null; // Presence test. if (empty($op) && empty($val) && !isset($data[$attr])) { return false; } // Empty attribute = fail. if (!isset($data[$attr])) { return false; } $prop = isset($data[$attr]) ? $data[$attr] : null; // Pattern matches and other operators. if ($op === '=' && $val && $val[0] === '/') { if (!preg_match($val, $prop)) { return false; } } elseif ( ($op === '=' && $prop != $val) || ($op === '!=' && $prop == $val) || ($op === '>' && $prop <= $val) || ($op === '<' && $prop >= $val) || ($op === '>=' && $prop < $val) || ($op === '<=' && $prop > $val) ) { return false; } } return true; } /** * Insert $values into an array with the given $path. * * @param array $data The data to insert into. * @param string $path The path to insert at. * @param mixed $values The values to insert. * @return array The data with $values inserted. */ public static function insert(array $data, $path, $values = null) { $tokens = explode('.', $path); if (strpos($path, '{') === false) { return self::_simpleOp('insert', $data, $tokens, $values); } $token = array_shift($tokens); $nextPath = implode('.', $tokens); foreach ($data as $k => $v) { if (self::_matchToken($k, $token)) { $data[$k] = self::insert($v, $nextPath, $values); } } return $data; } /** * Perform a simple insert/remove operation. * * @param string $op The operation to do. * @param array $data The data to operate on. * @param array $path The path to work on. * @param mixed $values The values to insert when doing inserts. * @return array $data. */ protected static function _simpleOp($op, $data, $path, $values = null) { $_list =& $data; $count = count($path); foreach ($path as $i => $key) { if (is_numeric($key) && intval($key) > 0 || $key === '0') { $key = intval($key); } if ($op === 'insert') { if ($i === $count - 1 && is_array($_list)) { $_list[$key] = $values; } else { if (!isset($_list[$key])) { $_list[$key] = array(); } $_list =& $_list[$key]; } if (!is_array($_list)) { return array(); } } elseif ($op === 'remove') { if ($i === count($path) - 1) { unset($_list[$key]); } else { if (!isset($_list[$key])) { return $data; } $_list =& $_list[$key]; } } } return $data; } /** * Remove data matching $path from the $data array. * * @param array $data The data to operate on * @param string $path A path expression to use to remove. * @return array The modified array. */ public static function remove(array $data, $path) { $tokens = explode('.', $path); if (strpos($path, '{') === false) { return self::_simpleOp('remove', $data, $tokens); } $token = array_shift($tokens); $nextPath = implode('.', $tokens); foreach ($data as $k => $v) { $match = self::_matchToken($k, $token); if ($match && is_array($v)) { $data[$k] = self::remove($v, $nextPath); } elseif ($match) { unset($data[$k]); } } return $data; } public static function combine(array $data, $keyPath, $valuePath = null) { } /** * Determines if one array contains the exact keys and values of another. * * @param array $data The data to search through. * @param array $needle The values to file in $data * @return boolean true if $data contains $needle, false otherwise * @link http://book.cakephp.org/2.0/en/core-utility-libraries/set.html#Set::contains */ public static function contains(array $data, array $needle) { if (empty($data) || empty($needle)) { return false; } $stack = array(); $i = 1; while (!empty($needle)) { $key = key($needle); $val = $needle[$key]; unset($needle[$key]); if (isset($data[$key]) && is_array($val)) { $next = $data[$key]; unset($data[$key]); if (!empty($val)) { $stack[] = array($val, $next); } } elseif (!isset($data[$key]) || $data[$key] != $val) { return false; } if (empty($needle) && !empty($stack)) { list($needle, $data) = array_pop($stack); } } return true; } /** * Test whether or not a given path exists in $data. * This method uses the same path syntax as Set2::extract() * * Checking for paths that could target more than one element will * make sure that at least one matching element exists. * * @param array $data The data to check. * @param string $path The path to check for. * @return boolean Existence of path. * @see Set2::extract() */ public static function check(array $data, $path) { $results = self::extract($data, $path); if (!is_array($results)) { return false; } return count($results) > 0; } /** * Recursively filters empty elements out of a route array, excluding '0'. * * @param array $data Either an array to filter, or value when in callback * @return array Filtered array * @link http://book.cakephp.org/2.0/en/core-utility-libraries/set.html#Set::filter */ public static function filter(array $data) { foreach ($data as $k => $v) { if (is_array($v)) { $data[$k] = Set2::filter($v); } } return array_filter($data, array('Set2', '_filter')); } /** * Callback function for filtering. * * @param array $var Array to filter. * @return boolean */ protected static function _filter($var) { if ($var === 0 || $var === '0' || !empty($var)) { return true; } return false; } /** * Collapses a multi-dimensional array into a single dimension, using a delimited array path for * each array element's key, i.e. array(array('Foo' => array('Bar' => 'Far'))) becomes * array('0.Foo.Bar' => 'Far').) * * @param array $data Array to flatten * @param string $separator String used to separate array key elements in a path, defaults to '.' * @return array * @link http://book.cakephp.org/2.0/en/core-utility-libraries/set.html#Set::flatten */ public static function flatten(array $data, $separator = '.') { $result = array(); $stack = array(); $path = null; reset($data); while (!empty($data)) { $key = key($data); $element = $data[$key]; unset($data[$key]); if (is_array($element)) { if (!empty($data)) { $stack[] = array($data, $path); } $data = $element; $path .= $key . $separator; } else { $result[$path . $key] = $element; } if (empty($data) && !empty($stack)) { list($data, $path) = array_pop($stack); } } return $result; } /** * This function can be thought of as a hybrid between PHP's `array_merge` and `array_merge_recursive`. * * The difference between this method and the built-in ones, is that if an array key contains another array, then * Set2::merge() will behave in a recursive fashion (unlike `array_merge`). But it will not act recursively for * keys that contain scalar values (unlike `array_merge_recursive`). * * Note: This function will work with an unlimited amount of arguments and typecasts non-array parameters into arrays. * * @param array $data Array to be merged * @param mixed $merge Array to merge with. The argument and all trailing arguments will be array cast when merged * @return array Merged array * @link http://book.cakephp.org/2.0/en/core-utility-libraries/set.html#Set::merge */ public static function merge(array $data, $merge) { $args = func_get_args(); $return = current($args); while (($arg = next($args)) !== false) { foreach ((array)$arg as $key => $val) { if (!empty($return[$key]) && is_array($return[$key]) && is_array($val)) { $return[$key] = self::merge($return[$key], $val); } elseif (is_int($key)) { $return[] = $val; } else { $return[$key] = $val; } } } return $return; } /** * Checks to see if all the values in the array are numeric * * @param array $array The array to check. * @return boolean true if values are numeric, false otherwise * @link http://book.cakephp.org/2.0/en/core-utility-libraries/set.html#Set::numeric */ public static function numeric(array $data) { if (empty($data)) { return false; } $values = array_values($data); $str = implode('', $values); return (bool) ctype_digit($str); } /** * Counts the dimensions of an array. * Only considers the dimension of the first element in the array. * * If you have an un-even or hetrogenous array, consider using Set2::maxDimensions() * to get the dimensions of the array. * * @param array $array Array to count dimensions on * @return integer The number of dimensions in $data * @link http://book.cakephp.org/2.0/en/core-utility-libraries/set.html#Set::countDim */ public static function dimensions(array $data) { if (empty($data)) { return 0; } reset($data); $depth = 1; while ($elem = array_shift($data)) { if (is_array($elem)) { $depth += 1; $data =& $elem; } else { break; } } return $depth; } /** * Counts the dimensions of *all* array elements. Useful for finding the maximum * number of dimensions in a mixed array. * * @param array $data Array to count dimensions on * @return integer The maximum number of dimensions in $data * @link http://book.cakephp.org/2.0/en/core-utility-libraries/set.html#Set::countDim */ public static function maxDimensions(array $data) { $depth = array(); if (is_array($data) && reset($data) !== false) { foreach ($data as $value) { $depth[] = Set2::dimensions((array)$value) + 1; } } return max($depth); } /** * Map a callback across all elements in a set. * Can be provided a path to only modify slices of the set. * */ public static function map(array $data, $path, $function = null) { } /** * Sorts an array by any value, determined by a Set-compatible path * * @param array $data An array of data to sort * @param string $path A Set-compatible path to the array value * @param string $dir Direction of sorting - either ascending (ASC), or descending (DESC) * @return array Sorted array of data * @link http://book.cakephp.org/2.0/en/core-utility-libraries/set.html#Set::sort */ public static function sort(array $data, $path, $dir) { $originalKeys = array_keys($data); if (is_numeric(implode('', $originalKeys))) { $data = array_values($data); } $sortValues = self::extract($data, $path); $sortCount = count($sortValues); $dataCount = count($data); // Make sortValues match the data length, as some keys could be missing // the sorted value path. if ($sortCount < $dataCount) { $sortValues = array_pad($sortValues, $dataCount, null); } $result = self::_squash($sortValues); $keys = self::extract($result, '{n}.id'); $values = self::extract($result, '{n}.value'); $dir = strtolower($dir); if ($dir === 'asc') { $dir = SORT_ASC; } elseif ($dir === 'desc') { $dir = SORT_DESC; } array_multisort($values, $dir, $keys, $dir); $sorted = array(); $keys = array_unique($keys); foreach ($keys as $k) { $sorted[] = $data[$k]; } return $sorted; } /** * Helper method for sort() * Sqaushes an array to a single hash so it can be sorted. * * @param array $data The data to squash. * @param string $key The key for the data. * @return array */ protected static function _squash($data, $key = null) { $stack = array(); foreach ($data as $k => $r) { $id = $k; if (!is_null($key)) { $id = $key; } if (is_array($r) && !empty($r)) { $stack = array_merge($stack, self::_squash($r, $id)); } else { $stack[] = array('id' => $id, 'value' => $r); } } return $stack; } /** * Computes the difference between two complex arrays. * This method differs from the built-in array_diff() in that it will preserve keys * and work on multi-dimensional arrays. * * @param mixed $data First value * @param mixed $data2 Second value * @return array Returns the key => value pairs that are not common in $data and $data2 * The expression for this function is ($data - $data2) + ($data2 - ($data - $data2)) * @link http://book.cakephp.org/2.0/en/core-utility-libraries/set.html#Set::diff */ public static function diff(array $data, $data2) { if (empty($data)) { return (array)$data2; } if (empty($data2)) { return (array)$data; } $intersection = array_intersect_key($data, $data2); while (($key = key($intersection)) !== null) { if ($data[$key] == $data2[$key]) { unset($data[$key]); unset($data2[$key]); } next($intersection); } return $data + $data2; } /** * Normalizes an array, and converts it to a standard format. * * @param mixed $data List to normalize * @param boolean $assoc If true, $data will be converted to an associative array. * @return array * @link http://book.cakephp.org/2.0/en/core-utility-libraries/set.html#Set::normalize */ public static function normalize(array $data, $assoc = true) { $keys = array_keys($data); $count = count($keys); $numeric = true; if (!$assoc) { for ($i = 0; $i < $count; $i++) { if (!is_int($keys[$i])) { $numeric = false; break; } } } if (!$numeric || $assoc) { $newList = array(); for ($i = 0; $i < $count; $i++) { if (is_int($keys[$i])) { $newList[$data[$keys[$i]]] = null; } else { $newList[$keys[$i]] = $data[$keys[$i]]; } } $data = $newList; } return $data; } }