praiadeseselle/wire/core/PagesRaw.php

1040 lines
28 KiB
PHP
Raw Normal View History

2022-03-08 15:55:41 +01:00
<?php namespace ProcessWire;
/**
* ProcessWire Pages Raw Tools
*
* ProcessWire 3.x, Copyright 2021 by Ryan Cramer
* https://processwire.com
*
*/
class PagesRaw extends Wire {
/**
* @var Pages
*
*/
protected $pages;
/**
* Construct
*
* @param Pages $pages
*
*/
public function __construct(Pages $pages) {
$this->pages = $pages;
}
/**
* Find pages and return raw data from them in a PHP array
*
* @param string|array|Selectors $selector
* @param string|Field|int|array $field Field/property name to get or array of them (or omit to get all)
* - Optionally use associative array to rename fields in returned value, i.e. `['title' => 'label']` returns 'title' as 'label' in return value.
* @param array $options See options for Pages::find
* - `objects` (bool): Use objects rather than associative arrays? (default=false)
* - `entities` (bool|array): Entity encode string values? True, or specify array of field names. (default=false)
* - `indexed` (bool): Index by page ID? (default=true)
* @return array
* @since 3.0.172
*
*/
public function find($selector, $field = '', $options = array()) {
if(!is_array($options)) $options = array('indexed' => (bool) $options);
$finder = new PagesRawFinder($this->pages);
$this->wire($finder);
return $finder->find($selector, $field, $options);
}
/**
* Get page (no exclusions) and return raw data from it in a PHP array
*
* @param string|array|Selectors $selector
* @param string|Field|int|array $field Field/property name to get or array of them (or omit to get all)
* @param array|bool $options See options for Pages::find
* - `objects` (bool): Use objects rather than associative arrays? (default=false)
* - `entities` (bool|array): Entity encode string values? True, or specify array of field names. (default=false)
* - `indexed` (bool): Index by page ID? (default=false)
* @return array
* @since 3.0.172
*
*/
public function get($selector, $field = '', $options = array()) {
if(!is_array($options)) $options = array('indexed' => (bool) $options);
$options['findOne'] = true;
if(!isset($options['findAll'])) $options['findAll'] = true;
$values = $this->find($selector, $field, $options);
return reset($values);
}
}
/**
* ProcessWire Pages Raw Finder
*
*/
class PagesRawFinder extends Wire {
/**
* @var Pages
*
*/
protected $pages;
/**
* @var array
*
*/
protected $options = array();
/**
* @var array
*
*/
protected $defaults = array(
'indexed' => true,
'objects' => false,
'entities' => false,
'findOne' => false,
);
/**
* @var string|array|Selectors
*
*/
protected $selector = '';
/**
* @var bool
*
*/
protected $selectorIsPageIDs = false;
/**
* @var array
*
*/
protected $requestFields = array();
/**
* @var array
*
*/
protected $nativeFields = array();
/**
* @var array
*
*/
protected $parentFields = array();
/**
* @var array
*
*/
protected $childrenFields = array();
/**
* @var array
*
*/
protected $customFields = array();
/**
* @var array
*
*/
protected $runtimeFields = array();
/**
* Fields to rename in returned value, i.e. [ 'title' => 'label' ]
*
* @var array
*
*/
protected $renameFields = array();
/**
* @var array
*
*/
protected $customCols = array();
/**
* Columns requested as fieldName.col rather than fieldName[col]
*
* (not currently accounted for, future use)
*
* @var array
*
*/
protected $customDotCols = array();
/**
* Results of the raw find
*
* @var array
*
*/
protected $values = array();
/**
* True to return array indexed by field name for each page, false to return single value for each page
*
* @var bool
*
*/
protected $getMultiple = true;
/**
* Get all data for pages?
*
* @var bool
*
*/
protected $getAll = false;
/**
* Get/join the pages_paths table?
*
* @var bool
*
*/
protected $getPaths = false;
/**
* IDs of pages to find, becomes array once known
*
* @var null|array
*
*/
protected $ids = null;
/**
* Construct
*
* @param Pages $pages
*
*/
public function __construct(Pages $pages) {
$this->pages = $pages;
}
/**
* @param string|int|array|Selectors
* @param string|array|Field $field
* @param array $options
*
*/
protected function init($selector, $field, $options) {
$fields = $this->wire()->fields;
$selectorString = '';
$this->selector = $selector;
$this->options = array_merge($this->defaults, $options);
$this->values = array();
$this->requestFields = array();
$this->customFields = array();
$this->nativeFields = array();
$this->customCols = array();
$this->getMultiple = true;
$this->getAll = false;
$this->ids = null;
if(is_array($selector)) {
$val = reset($selector);
$key = key($selector);
if(ctype_digit("$key") && ctype_digit("$val")) $this->selectorIsPageIDs = true;
} else {
$selectorString = (string) $selector;
}
if(empty($field) && !$this->selectorIsPageIDs) {
// check if field specified in selector instead
$field = array();
$multi = false;
if(!$selector instanceof Selectors) {
$selector = new Selectors($selector);
$this->wire($selector);
}
foreach($selector as $item) {
if(!$item instanceof SelectorEqual) continue;
$name = $item->field();
if($name !== 'field' && $name !== 'join' && $name !== 'fields') continue;
if($name !== 'fields' && $fields->get($name)) continue;
$value = $item->value;
if(is_array($value)) {
$field = array_merge($field, $value);
} else if($value === 'all') {
$this->getAll = true;
} else {
$field[] = $value;
}
$selector->remove($item);
if($name === 'fields') $multi = true;
}
$this->selector = $selector;
if(!$multi && count($field) === 1) $field = reset($field);
}
if(empty($field)) {
$this->getAll = true;
} else if(is_string($field) && strpos($field, ',') !== false) {
// multiple fields requested in CSV string, we will return an array for each page
$this->requestFields = explode(',', $field);
} else if(is_array($field)) {
// one or more fields requested in array, we will return an array for each page
$requestFields = array();
$renameFields = array();
foreach($field as $key => $value) {
if(is_string($key) && !ctype_digit($key) && !is_array($value)) {
$requestFields[] = $key;
$renameFields[$key] = $value;
} else {
$requestFields[] = $value;
}
}
$this->requestFields = $requestFields;
$this->renameFields = $renameFields;
} else {
// one field requested in string or Field object
$this->requestFields = array($field);
$this->getMultiple = false;
}
if($this->getAll) {
if($this->wire()->modules->isInstalled('PagePaths')) {
$this->getPaths = true;
$this->runtimeFields['url'] = 'url';
$this->runtimeFields['path'] = 'path';
}
} else {
// split request fields into nativeFields and customFields
$this->splitFields();
}
// detect 'objects' and 'entities' options in selector
$optionsValues = array();
foreach(array('objects', 'entities', 'options') as $name) {
if($this->selectorIsPageIDs) continue;
if($selectorString && strpos($selectorString, "$name=") === false) continue;
if($fields->get($name)) continue; // if maps to a real field then ignore
$result = Selectors::selectorHasField($this->selector, $name, array(
'operator' => '=',
'verbose' => true,
'remove' => true,
));
$value = $result['value'];
if($result['result'] && $value && !isset($options[$name])) {
if($name === 'options') {
if(is_string($value)) $optionsValues[] = $value;
if(is_array($value)) $optionsValues = array_merge($optionsValues, $value);
} else if(is_array($value)) {
$this->options[$name] = array();
foreach($value as $v) $this->options[$name][$v] = $v;
} else if(!ctype_digit("$value")) {
$this->options[$name] = $value;
} else {
$this->options[$name] = (bool) ((int) $value);
}
}
if(!empty($result['selectors'])) $this->selector = $result['selectors'];
}
foreach(array('objects', 'entities') as $name) {
if(in_array($name, $optionsValues)) $this->options[$name] = true;
}
}
/**
* Find pages and return raw data from them in a PHP array
*
* How to use the `$field` argument:
*
* - If you provide an array for $field then it will return an array for each page, indexed by
* the field names you requested.
*
* - If you provide a string (field name) or Field object, then it will return an array with
* the values of the 'data' column of that field.
*
* - You may request field name(s) like `field.subfield` to retrieve a specific column/subfield.
*
* - You may request field name(s) like `field.*` to return all columns/subfields for `field`,
* in this case, an associative array value will be returned for each page.
*
* - If you specify an associative array for the $field argument, you can optionally rename
* fields in returned value. For example, if you wanted to get the 'title' field but return
* it as a field named 'headline' in the return value, you would specify the array
* `[ 'title' => 'headline' ]` for the $field argument. (3.0.176+)
*
* @param string|array|Selectors $selector
* @param string|Field|int|array $field Field/property name or array of of them
* @param array $options See options for Pages::find
* @return array
* @since 3.0.172
*
*/
public function find($selector, $field = '', $options = array()) {
static $level = 0;
$level++;
$this->init($selector, $field, $options);
// requested native pages table fields/properties
if(count($this->nativeFields) || $this->getAll || $this->getPaths) {
// one or more native pages table column(s) requested
$this->findNativeFields();
}
// requested custom fields
if(count($this->customFields) || $this->getAll) {
$this->findCustom();
}
// reduce return value when expected
if(!$this->getMultiple) {
foreach($this->values as $id => $row) {
$this->values[$id] = reset($row);
}
}
if(!$this->options['indexed']) {
$this->values = array_values($this->values);
}
if(count($this->renameFields)) {
$this->renames($this->values);
}
if($this->options['entities']) {
if($this->options['objects'] || $level === 1) {
$this->entities($this->values);
}
}
if($this->options['objects']) {
$this->objects($this->values);
}
$level--;
return $this->values;
}
/**
* Split requestFields into native and custom field arrays
*
* Populates $this->nativeFields, $this->customFields, $this->customCols
*
*/
protected function splitFields() {
$fields = $this->wire()->fields;
$fails = array();
// split request fields into custom fields and native (pages table) fields
foreach($this->requestFields as $key => $fieldName) {
if(empty($fieldName)) continue;
if(is_string($fieldName)) $fieldName = trim($fieldName);
$colName = '';
$dotCol = false;
$fieldObject = null;
$fullName = $fieldName;
if($fieldName === '*') {
// get all (not yet supported)
} else if($fieldName instanceof Field) {
// Field object
$fieldObject = $fieldName;
} else if(is_array($fieldName)) {
// Array where [ 'field' => [ 'subfield' ]]
$colName = $fieldName; // array
$fieldName = $key;
if($fieldName === 'parent' || $fieldName === 'children') {
// passthru
} else {
$fieldObject = isset($this->customFields[$fieldName]) ? $this->customFields[$fieldName] : null;
if(!$fieldObject) $fieldObject = $fields->get($fieldName);
if(!$fieldObject) continue;
}
} else if(is_int($fieldName) || ctype_digit("$fieldName")) {
// Field ID
$fieldObject = $fields->get((int) $fieldName);
} else if(is_string($fieldName)) {
// Field name, subfield/column may optionally be specified as field.subfield
if(strpos($fieldName, '.')) {
list($fieldName, $colName) = explode('.', $fieldName, 2);
$dotCol = true;
} else if(strpos($fieldName, '[')) {
list($fieldName, $colName) = explode('[', $fieldName, 2);
$colName = rtrim($colName, ']');
}
if($fieldName === 'parent' || $fieldName === 'children') {
// passthru
} else {
$fieldObject = isset($this->customFields[$fieldName]) ? $this->customFields[$fieldName] : null;
if(!$fieldObject) $fieldObject = $fields->get($fieldName);
}
} else {
// something we do not recognize
$fails[] = $fieldName;
continue;
}
if($fieldName === 'parent') {
// @todo not yet supported
$this->parentFields[$fullName] = array($fieldName, $colName);
} else if($fieldName === 'children') {
// @todo not yet supported
$this->childrenFields[$fullName] = array($fieldName, $colName);
} else if($fullName === 'url' || $fullName === 'path') {
if($this->wire()->modules->isInstalled('PagePaths')) {
$this->runtimeFields[$fullName] = $fullName;
$this->getPaths = true;
} else {
$fails[] = "Property '$fullName' requires the PagePaths module be installed";
}
} else if($fieldObject instanceof Field) {
$this->customFields[$fieldName] = $fieldObject;
if(!empty($colName)) {
$colNames = is_array($colName) ? $colName : array($colName);
foreach($colNames as $col) {
if(!isset($this->customCols[$fieldName])) $this->customCols[$fieldName] = array();
$this->customCols[$fieldName][$col] = $col;
if($dotCol) {
if(!isset($this->customDotCols[$fieldName])) $this->customDotCols[$fieldName] = array();
$this->customDotCols[$fieldName][$col] = $col;
}
}
}
} else {
$this->nativeFields[$fieldName] = $fieldName;
}
}
if(count($fails)) $this->unknownFieldsException($fails);
}
/**
* Find raw native fields
*
*/
protected function findNativeFields() {
$this->ids = array();
$allNatives = array();
$fails = array();
$rootUrl = $this->wire()->config->urls->root;
$templates = $this->wire()->templates;
$templatesById = array();
$getPaths = $this->getPaths;
foreach($this->findIDs($this->selector, '*') as $row) {
$id = (int) $row['id'];
$this->ids[$id] = $id;
$this->values[$id] = isset($this->values[$id]) ? array_merge($this->values[$id], $row) : $row;
if(empty($allNatives)) {
foreach(array_keys($row) as $key) {
$allNatives[$key] = $key;
}
}
}
if(!count($this->values)) return;
if($this->getAll) $this->nativeFields = $allNatives;
// native columns we will populate into $values
$getNatives = array();
foreach($this->nativeFields as $fieldName) {
if($fieldName === '*' || $fieldName === 'pages' || $fieldName === 'pages.*') {
// get all columns
$colName = '';
} else if(strpos($fieldName, 'pages.') === 0) {
// pages table column requested by name
list(,$colName) = explode('.', $fieldName, 2);
} else {
// column requested by name on its own
$colName = $fieldName;
}
if(empty($colName)) {
// get all native pages table columns
$getNatives = $allNatives;
} else if(isset($allNatives[$colName])) {
// get specific native pages table columns
$getNatives[$colName] = $colName;
} else {
// fieldName is not a known field or pages column
$fails[] = "$fieldName";
}
}
if(count($fails)) $this->unknownFieldsException($fails, 'column/field');
if(!count($getNatives) && !$getPaths) return;
// remove any native data that is present but was not requested and populate any runtime fields
foreach($this->values as $id => $row) {
$templateId = (int) $row['templates_id'];
foreach($row as $colName => $value) {
if($getPaths && $colName === 'path') {
// populate path and/or url runtime properties
if(!isset($templatesById[$templateId])) $templatesById[$templateId] = $templates->get($templateId);
$template = $templatesById[$templateId]; /** @var Template $template */
$slash = $template->slashUrls ? '/' : '';
$path = strlen($value) && $value !== '/' ? "$value$slash" : '';
if(isset($this->runtimeFields['url'])) {
$this->values[$id]['url'] = $rootUrl . $path;
}
if(isset($this->runtimeFields['path'])) {
$this->values[$id]['path'] = "/$path";
} else {
unset($this->values[$id]['path']);
}
} else if(!isset($getNatives[$colName])) {
unset($this->values[$id][$colName]);
}
}
}
}
/**
* Gateway to finding custom fields whether specific, all or none
*
*/
protected function findCustom() {
if(count($this->customFields)) {
// one or more custom fields requested
if($this->ids === null) {
// only find IDs if we didnt already in the nativeFields section
$this->ids = $this->findIDs($this->selector, false);
}
if(!count($this->ids)) return;
foreach($this->customFields as $fieldName => $field) {
/** @var Field $field */
$cols = isset($this->customCols[$fieldName]) ? $this->customCols[$fieldName] : array();
$this->findCustomField($field, $cols);
}
} else if($this->getAll && !empty($this->ids)) {
$this->findCustomAll();
}
}
/**
* Find raw custom field
*
* @param Field $field
* @param array $cols
* @throws WireException
*
*/
protected function findCustomField(Field $field, array $cols) {
$database = $this->wire()->database;
$sanitizer = $this->wire()->sanitizer;
$getArray = true;
$getCols = array();
$skipCols = array();
$getAllCols = false;
$pageRefCols = array();
/** @var FieldtypeMulti $fieldtypeMulti */
$fieldtype = $field->type;
$fieldtypeMulti = $field->type instanceof FieldtypeMulti ? $fieldtype : null;
$fieldtypePage = $fieldtype instanceof FieldtypePage ? $fieldtype : null;
$fieldtypeRepeater = $fieldtype instanceof FieldtypeRepeater ? $fieldtype : null;
$fieldName = $field->name;
$schema = $fieldtype->getDatabaseSchema($field);
$schema = $fieldtype->trimDatabaseSchema($schema, array('trimDefault' => false));
$table = $database->escapeTable($field->getTable());
$sorts = array();
if(empty($table)) return;
if(empty($cols)) {
// no cols specified
$trimSchema = $fieldtype->trimDatabaseSchema($schema, array('trimDefault' => true, 'trimMeta' => true));
unset($trimSchema['data']);
foreach($trimSchema as $key => $value) {
// multi-language columns do not count as custom schema
if(strpos($key, 'data') === 0 && ctype_digit(substr($key, 4))) unset($trimSchema[$key]);
}
if(empty($trimSchema)) {
// if table doesnt maintain a custom schema, just get data column
$getArray = false;
$getCols[] = 'data';
} else {
// table maintains custom schema, get all columns
$getAllCols = true;
$skipCols[] = 'pages_id';
}
} else if(reset($cols) === '*') {
$getAllCols = true;
} else {
foreach($cols as $key => $col) {
$col = $sanitizer->fieldName($col);
$col = $database->escapeCol($col);
if(isset($schema[$col])) {
$getCols[$col] = $col;
} else if($fieldtypePage || $fieldtypeRepeater) {
$pageRefCols[$col] = $col;
} else {
// unknown column
}
}
if(count($pageRefCols)) {
// get just the data column when a field within a Page reference is asked for
$getCols['data'] = 'data';
}
if(count($getCols) === 1 && !$this->getMultiple) {
// if only getting single field we will populate its value rather than
// its value in an associative array
$getArray = false;
}
}
if($fieldtypeMulti) {
$orderByCols = $fieldtypeMulti->get('orderByCols');
if($fieldtypeMulti->useOrderByCols && !empty($orderByCols)) {
foreach($orderByCols as $key => $col) {
$desc = strpos($col, '-') === 0 ? ' DESC' : '';
$col = $sanitizer->fieldName(ltrim($col, '-'));
if(!array_key_exists($col, $schema)) continue;
$sorts[$key] = '`' . $database->escapeCol($col) . '`' . $desc;
}
}
if(empty($sorts) && isset($schema['sort'])) {
$sorts[] = "`sort`";
}
}
$this->ids(true); // converts this->ids to CSV string
$idsCSV = &$this->ids;
$colSQL = $getAllCols ? '*' : '`' . implode('`,`', $getCols) . '`';
if(!$getAllCols && !in_array('pages_id', $getCols)) $colSQL .= ',`pages_id`';
$orderby = array();
if(!count($this->nativeFields)) $orderby[] = "FIELD(pages_id, $idsCSV)";
if(count($sorts)) $orderby[] = implode(',', $sorts);
$sql = "SELECT $colSQL FROM `$table` WHERE pages_id IN($idsCSV) ";
if(count($orderby)) $sql .= "ORDER BY " . implode(',', $orderby);
$query = $database->prepare($sql);
$query->execute();
while($row = $query->fetch(\PDO::FETCH_ASSOC)) {
$id = $row['pages_id'];
if(!$getAllCols && !isset($getCols['pages_id'])) unset($row['pages_id']);
foreach($skipCols as $skipCol) {
unset($row[$skipCol]);
}
if($getAllCols) {
$value = $row;
} else if($getArray) {
$value = array();
foreach($getCols as $col) {
$value[$col] = isset($row[$col]) ? $row[$col] : null;
}
} else {
$col = reset($getCols);
if(empty($col)) $col = 'data';
$value = $row[$col];
}
if(!isset($this->values[$id])) {
// Overall page placeholder array
$this->values[$id] = array();
}
if($fieldtypeMulti) {
// FieldtypeMulti types may contain multiple rows
/** @var FieldtypeMulti $fieldtype */
if(!isset($this->values[$id][$fieldName])) {
$this->values[$id][$fieldName] = array();
}
if($fieldtypePage && count($pageRefCols)) {
// reduce page reference to just the IDs, indexed by IDs
if(isset($value['data'])) $value = $value['data'];
$this->values[$id][$fieldName][$value] = $value;
} else {
$this->values[$id][$fieldName][] = $value;
}
} else if($fieldtypeRepeater && count($pageRefCols)) {
$repeaterIds = isset($value['data']) ? explode(',', $value['data']) : explode(',', $value);
foreach($repeaterIds as $repeaterId) {
$this->values[$id][$fieldName][$repeaterId] = $repeaterId;
}
} else {
$this->values[$id][$fieldName] = $value;
}
}
$query->closeCursor();
if(count($pageRefCols)) {
if($fieldtypePage) {
$this->findCustomFieldtypePage($field, $fieldName, $pageRefCols);
} else if($fieldtypeRepeater) {
$this->findCustomFieldtypePage($field, $fieldName, $pageRefCols);
}
}
}
protected function findCustomFieldtypePage(Field $field, $fieldName, array $pageRefCols) {
// print_r($values);
$pageRefIds = array();
foreach($this->values as $pageId => $row) {
if(!isset($row[$fieldName])) continue;
$pageRefIds = array_merge($pageRefIds, $row[$fieldName]);
}
if(!$this->getMultiple && count($pageRefCols) === 1) {
$pageRefCols = implode('', $pageRefCols);
}
$pageRefIds = array_unique($pageRefIds);
$finder = new PagesRawFinder($this->pages);
$this->wire($finder);
$options = $this->options;
$options['indexed'] = true;
$pageRefRows = $finder->find($pageRefIds, $pageRefCols, $options);
foreach($this->values as $pageId => $pageRow) {
if(!isset($pageRow[$fieldName])) continue;
foreach($pageRow[$fieldName] as $pageRefId) {
$this->values[$pageId][$fieldName][$pageRefId] = $pageRefRows[$pageRefId];
}
if(!$this->getMultiple && $field->get('derefAsPage') > 0) {
$this->values[$pageId][$fieldName] = reset($this->values[$pageId][$fieldName]);
} else if(empty($this->options['indexed'])) {
$this->values[$pageId][$fieldName] = array_values($this->values[$pageId][$fieldName]);
}
}
}
/**
* Find/populate all custom fields
*
*/
protected function findCustomAll() {
$idsByTemplate = array();
foreach($this->ids() as $id) {
if(!isset($this->values[$id])) continue;
$row = $this->values[$id];
$templateId = $row['templates_id'];
if(!isset($idsByTemplate[$templateId])) $idsByTemplate[$templateId] = array();
$idsByTemplate[$templateId][$id] = $id;
}
foreach($idsByTemplate as $templateId => $pageIds) {
$template = $this->wire()->templates->get($templateId);
if(!$template) continue;
foreach($template->fieldgroup as $field) {
$this->findCustomField($field, array());
}
}
}
/**
* Front-end to pages.findIDs that optionally accepts array of page IDs
*
* @param array|string|Selectors $selector
* @param bool|string $verbose One of true, false, or '*'
* @param array $options
* @return array
* @throws WireException
*
*/
protected function findIDs($selector, $verbose, array $options = array()) {
$options = array_merge($this->options, $options);
$options['verbose'] = $verbose;
$options['indexed'] = true;
$options['joinPath'] = $this->getPaths;
// if selector was just a page ID, return it in an id indexed array
if(is_int($selector) || (is_string($selector) && ctype_digit($selector))) {
$id = (int) $selector;
return array($id => $id);
}
// if selector is not array of page IDs then let pages.findIDs handle it
if(!is_array($selector) || !isset($selector[0]) || !ctype_digit((string) $selector[0])) {
return $this->pages->findIDs($selector, $options);
}
// at this point selector is an array of page IDs
if(empty($verbose)) {
// if selector already has what is needed and verbose data not needed,
// then return it now, but make sure it is indexed by ID first
$a = array();
foreach($selector as $id) $a[(int) $id] = (int) $id;
return $a;
}
// convert selector to CSV string of page IDs
$selector = implode(',', array_map('intval', $selector));
$selects = array();
$joins = array();
$wheres = array("id IN($selector)");
if($verbose === '*') {
// get all columns
$selects[] = 'pages.*';
} else {
// get just base columns
$selects = array('pages.id', 'pages.templates_id', 'pages.parent_id');
}
if($this->getPaths) {
$selects[] = 'pages_paths.path AS path';
$joins[] = 'LEFT JOIN pages_paths ON pages_paths.pages_id=pages.id';
}
$sql =
"SELECT " . implode(', ', $selects) . " " .
"FROM pages " .
(count($joins) ? implode(' ', $joins) . " " : '') .
"WHERE " . implode(' ', $wheres);
$query = $this->wire()->database->prepare($sql);
$query->execute();
$rows = array();
while($row = $query->fetch(\PDO::FETCH_ASSOC)) {
$id = (int) $row['id'];
$rows[$id] = $row;
}
$query->closeCursor();
return $rows;
}
/**
* Convert associative arrays to objects
*
* @param array $values
*
*/
protected function objects(&$values) {
foreach(array_keys($values) as $key) {
$value = $values[$key];
if(!is_array($value)) continue;
reset($value);
if(is_int(key($value))) continue;
$this->objects($value);
$values[$key] = (object) $value;
}
}
/**
* Apply entity encoding to all strings in given value, recursively
*
* @param mixed $value
*
*/
protected function entities(&$value) {
$prefix = ''; // populate for testing only
if(is_string($value)) {
// entity-encode
$value = $prefix . htmlentities($value, ENT_QUOTES, 'UTF-8');
} else if(is_array($value)) {
// iterate and go recursive
foreach(array_keys($value) as $key) {
if(is_array($value[$key])) {
$this->entities($value[$key]);
} else if(is_string($value[$key])) {
if($this->options['entities'] === true || $this->options['entities'] === $key || isset($this->options['entities'][$key])) {
$value[$key] = $prefix . htmlentities($value[$key], ENT_QUOTES, 'UTF-8');
}
}
}
} else {
// leave as-is
}
}
/**
* Rename fields on request
*
* @param array $values
* @since 3.0.167
*
*/
protected function renames(&$values) {
foreach($values as $key => $value) {
if(!is_array($value)) continue;
foreach($value as $k => $v) {
if(is_array($v)) $this->renames($v);
if(isset($this->renameFields[$k])) {
unset($values[$key][$k]);
$name = $this->renameFields[$k];
} else {
$name = $k;
}
$values[$key][$name] = $v;
}
}
}
/**
* Get or convert $this->ids to/from CSV
*
* The point of this is just to minimize the quantity of copies of IDs we are keeping around.
* In case the quantity gets to be huge, it'll be more memory friendly.
*
* @param bool $csv
* @return array|string
*
*/
protected function ids($csv = false) {
if($this->ids === null) return $csv ? '' : array();
if($csv) {
if(is_array($this->ids)) $this->ids = implode(',', array_map('intval', $this->ids));
} else if(is_string($this->ids)) {
// this likely cannot occur with current logic but here in case that changes
$this->ids = explode(',', $this->ids);
}
return $this->ids;
}
protected function unknownFieldsException(array $fieldNames, $context = '') {
if($context) $context = " $context";
$s = "Unknown$context name(s) for findRaw: " . implode(', ', $fieldNames);
throw new WireException($s);
}
}