Initial commit
This commit is contained in:
commit
08a8a71c55
631 changed files with 139902 additions and 0 deletions
152
public/site/plugins/kql/src/Kql/Help.php
Normal file
152
public/site/plugins/kql/src/Kql/Help.php
Normal file
|
|
@ -0,0 +1,152 @@
|
|||
<?php
|
||||
|
||||
namespace Kirby\Kql;
|
||||
|
||||
use Kirby\Toolkit\A;
|
||||
use ReflectionClass;
|
||||
use ReflectionMethod;
|
||||
|
||||
/**
|
||||
* Providing help information about
|
||||
* queried objects, methods, arrays...
|
||||
*
|
||||
* @package Kirby KQL
|
||||
* @author Bastian Allgeier <bastian@getkirby.com>
|
||||
* @link https://getkirby.com
|
||||
* @copyright Bastian Allgeier
|
||||
* @license https://getkirby.com/license
|
||||
*/
|
||||
class Help
|
||||
{
|
||||
/**
|
||||
* Provides information about passed value
|
||||
* depending on its type
|
||||
*/
|
||||
public static function for($value): array
|
||||
{
|
||||
if (is_array($value) === true) {
|
||||
return static::forArray($value);
|
||||
}
|
||||
|
||||
if (is_object($value) === true) {
|
||||
return static::forObject($value);
|
||||
}
|
||||
|
||||
return [
|
||||
'type' => gettype($value),
|
||||
'value' => $value
|
||||
];
|
||||
}
|
||||
|
||||
/**
|
||||
* @internal
|
||||
*/
|
||||
public static function forArray(array $array): array
|
||||
{
|
||||
return [
|
||||
'type' => 'array',
|
||||
'keys' => array_keys($array),
|
||||
];
|
||||
}
|
||||
|
||||
/**
|
||||
* Gathers information for method about
|
||||
* name, parameters, return type etc.
|
||||
* @internal
|
||||
*/
|
||||
public static function forMethod(object $object, string $method): array
|
||||
{
|
||||
$reflection = new ReflectionMethod($object, $method);
|
||||
$returns = $reflection->getReturnType()?->getName();
|
||||
$params = [];
|
||||
|
||||
foreach ($reflection->getParameters() as $param) {
|
||||
$name = $param->getName();
|
||||
$required = $param->isOptional() === false;
|
||||
$type = $param->hasType() ? $param->getType()->getName() : null;
|
||||
$default = null;
|
||||
|
||||
if ($param->isDefaultValueAvailable()) {
|
||||
$default = $param->getDefaultValue();
|
||||
}
|
||||
|
||||
$call = '';
|
||||
|
||||
if ($type !== null) {
|
||||
$call = $type . ' ';
|
||||
}
|
||||
|
||||
$call .= '$' . $name;
|
||||
|
||||
if ($required === false && $default !== null) {
|
||||
$call .= ' = ' . var_export($default, true);
|
||||
}
|
||||
|
||||
$p['call'] = $call;
|
||||
|
||||
$params[$name] = compact('name', 'type', 'required', 'default', 'call');
|
||||
}
|
||||
|
||||
$call = '.' . $method;
|
||||
|
||||
if (empty($params) === false) {
|
||||
$call .= '(' . implode(', ', array_column($params, 'call')) . ')';
|
||||
}
|
||||
|
||||
return [
|
||||
'call' => $call,
|
||||
'name' => $method,
|
||||
'params' => $params,
|
||||
'returns' => $returns
|
||||
];
|
||||
}
|
||||
|
||||
/**
|
||||
* Gathers informations for each unique method
|
||||
* @internal
|
||||
*/
|
||||
public static function forMethods(object $object, array $methods): array
|
||||
{
|
||||
$methods = array_unique($methods);
|
||||
$reflection = [];
|
||||
|
||||
sort($methods);
|
||||
|
||||
foreach ($methods as $methodName) {
|
||||
if (method_exists($object, $methodName) === false) {
|
||||
continue;
|
||||
}
|
||||
|
||||
$reflection[$methodName] = static::forMethod($object, $methodName);
|
||||
}
|
||||
|
||||
return $reflection;
|
||||
}
|
||||
|
||||
/**
|
||||
* Retrieves info for objects either from Interceptor (to
|
||||
* only list allowed methods) or via reflection
|
||||
* @internal
|
||||
*/
|
||||
public static function forObject(object $object): array
|
||||
{
|
||||
// get interceptor object to only return info on allowed methods
|
||||
$interceptor = Interceptor::replace($object);
|
||||
|
||||
if ($interceptor instanceof Interceptor) {
|
||||
return $interceptor->__debugInfo();
|
||||
}
|
||||
|
||||
// for original classes, use reflection
|
||||
$class = new ReflectionClass($object);
|
||||
$methods = A::map(
|
||||
$class->getMethods(),
|
||||
fn ($method) => static::forMethod($object, $method->getName())
|
||||
);
|
||||
|
||||
return [
|
||||
'type' => $class->getName(),
|
||||
'methods' => $methods
|
||||
];
|
||||
}
|
||||
}
|
||||
295
public/site/plugins/kql/src/Kql/Interceptor.php
Normal file
295
public/site/plugins/kql/src/Kql/Interceptor.php
Normal file
|
|
@ -0,0 +1,295 @@
|
|||
<?php
|
||||
|
||||
namespace Kirby\Kql;
|
||||
|
||||
use Closure;
|
||||
use Kirby\Cms\App;
|
||||
use Kirby\Exception\InvalidArgumentException;
|
||||
use Kirby\Exception\PermissionException;
|
||||
use Kirby\Toolkit\Str;
|
||||
use ReflectionFunction;
|
||||
use ReflectionMethod;
|
||||
use Throwable;
|
||||
|
||||
/**
|
||||
* Base class for proxying core classes to
|
||||
* intercept method calls that are not allowed
|
||||
* on the related core class
|
||||
*
|
||||
* @package Kirby KQL
|
||||
* @author Bastian Allgeier <bastian@getkirby.com>
|
||||
* @link https://getkirby.com
|
||||
* @copyright Bastian Allgeier
|
||||
* @license https://getkirby.com/license
|
||||
*/
|
||||
abstract class Interceptor
|
||||
{
|
||||
public const CLASS_ALIAS = null;
|
||||
|
||||
protected $toArray = [];
|
||||
|
||||
public function __construct(protected $object)
|
||||
{
|
||||
}
|
||||
|
||||
/**
|
||||
* Magic caller that prevents access
|
||||
* to restricted methods
|
||||
*/
|
||||
public function __call(string $method, array $args = [])
|
||||
{
|
||||
if ($this->isAllowedMethod($method) === true) {
|
||||
return $this->object->$method(...$args);
|
||||
}
|
||||
|
||||
$this->forbiddenMethod($method);
|
||||
}
|
||||
|
||||
/**
|
||||
* Return information about corresponding object
|
||||
* incl. information about allowed methods
|
||||
*/
|
||||
public function __debugInfo(): array
|
||||
{
|
||||
$help = Help::forMethods($this->object, $this->allowedMethods());
|
||||
|
||||
return [
|
||||
'type' => $this::CLASS_ALIAS,
|
||||
'methods' => $help,
|
||||
'value' => $this->toArray()
|
||||
];
|
||||
}
|
||||
|
||||
/**
|
||||
* Returns list of allowed classes. Specific list
|
||||
* to be implemented in specific interceptor child classes.
|
||||
* @codeCoverageIgnore
|
||||
*/
|
||||
public function allowedMethods(): array
|
||||
{
|
||||
return [];
|
||||
}
|
||||
|
||||
/**
|
||||
* Returns class name for Interceptor that responds
|
||||
* to passed name string of a Kirby core class
|
||||
* @internal
|
||||
*/
|
||||
public static function class(string $class): string
|
||||
{
|
||||
return str_replace('Kirby\\', 'Kirby\\Kql\\Interceptors\\', $class);
|
||||
}
|
||||
|
||||
/**
|
||||
* Throws exception for accessing a restricted method
|
||||
* @throws \Kirby\Exception\PermissionException
|
||||
*/
|
||||
protected function forbiddenMethod(string $method)
|
||||
{
|
||||
$name = get_class($this->object) . '::' . $method . '()';
|
||||
throw new PermissionException('The method "' . $name . '" is not allowed in the API context');
|
||||
}
|
||||
|
||||
/**
|
||||
* Checks if method is allowed to call
|
||||
*/
|
||||
public function isAllowedMethod($method)
|
||||
{
|
||||
$kirby = App::instance();
|
||||
$name = strtolower(get_class($this->object) . '::' . $method);
|
||||
|
||||
// get list of blocked methods from config
|
||||
$blocked = $kirby->option('kql.methods.blocked', []);
|
||||
$blocked = array_map('strtolower', $blocked);
|
||||
|
||||
// check in the block list from the config
|
||||
if (in_array($name, $blocked) === true) {
|
||||
return false;
|
||||
}
|
||||
|
||||
// check in class allow list
|
||||
if (in_array($method, $this->allowedMethods()) === true) {
|
||||
return true;
|
||||
}
|
||||
|
||||
// get list of explicitly allowed methods from config
|
||||
$allowed = $kirby->option('kql.methods.allowed', []);
|
||||
$allowed = array_map('strtolower', $allowed);
|
||||
|
||||
// check in the allow list from the config
|
||||
if (in_array($name, $allowed) === true) {
|
||||
return true;
|
||||
}
|
||||
|
||||
// support for model methods with docblock comment
|
||||
if ($this->isAllowedCallable($method) === true) {
|
||||
return true;
|
||||
}
|
||||
|
||||
// support for custom methods with docblock comment
|
||||
if ($this->isAllowedCustomMethod($method) === true) {
|
||||
return true;
|
||||
}
|
||||
|
||||
return false;
|
||||
}
|
||||
|
||||
/**
|
||||
* Checks if closure or object method is allowed
|
||||
*/
|
||||
protected function isAllowedCallable($method): bool
|
||||
{
|
||||
try {
|
||||
$ref = match (true) {
|
||||
$method instanceof Closure
|
||||
=> new ReflectionFunction($method),
|
||||
is_string($method) === true
|
||||
=> new ReflectionMethod($this->object, $method),
|
||||
default
|
||||
=> throw new InvalidArgumentException('Invalid method')
|
||||
};
|
||||
|
||||
if ($comment = $ref->getDocComment()) {
|
||||
if (Str::contains($comment, '@kql-allowed') === true) {
|
||||
return true;
|
||||
}
|
||||
}
|
||||
} catch (Throwable) {
|
||||
return false;
|
||||
}
|
||||
|
||||
return false;
|
||||
}
|
||||
|
||||
protected function isAllowedCustomMethod(string $method): bool
|
||||
{
|
||||
// has no custom methods
|
||||
if (property_exists($this->object, 'methods') === false) {
|
||||
return false;
|
||||
}
|
||||
|
||||
// does not have that method
|
||||
if (!$call = $this->method($method)) {
|
||||
return false;
|
||||
}
|
||||
|
||||
// check for a docblock comment
|
||||
if ($this->isAllowedCallable($call) === true) {
|
||||
return true;
|
||||
}
|
||||
|
||||
return false;
|
||||
}
|
||||
|
||||
/**
|
||||
* Returns a registered method by name, either from
|
||||
* the current class or from a parent class ordered by
|
||||
* inheritance order (top to bottom)
|
||||
*/
|
||||
protected function method(string $method)
|
||||
{
|
||||
if (isset($this->object::$methods[$method]) === true) {
|
||||
return $this->object::$methods[$method];
|
||||
}
|
||||
|
||||
foreach (class_parents($this->object) as $parent) {
|
||||
if (isset($parent::$methods[$method]) === true) {
|
||||
return $parent::$methods[$method];
|
||||
}
|
||||
}
|
||||
|
||||
return null;
|
||||
}
|
||||
|
||||
/**
|
||||
* Tries to replace a Kirby core object with the
|
||||
* corresponding interceptor.
|
||||
* @throws \Kirby\Exception\InvalidArgumentException for non-objects
|
||||
* @throws \Kirby\Exception\PermissionException when accessing blocked class
|
||||
*/
|
||||
public static function replace($object)
|
||||
{
|
||||
if (is_object($object) === false) {
|
||||
throw new InvalidArgumentException('Unsupported value: ' . gettype($object));
|
||||
}
|
||||
|
||||
$kirby = App::instance();
|
||||
$class = get_class($object);
|
||||
$name = strtolower($class);
|
||||
|
||||
// 1. Is $object class explicitly blocked?
|
||||
// get list of blocked classes from config
|
||||
$blocked = $kirby->option('kql.classes.blocked', []);
|
||||
$blocked = array_map('strtolower', $blocked);
|
||||
|
||||
// check in the block list from the config
|
||||
if (in_array($name, $blocked) === true) {
|
||||
throw new PermissionException('Access to the class "' . $class . '" is blocked');
|
||||
}
|
||||
|
||||
// 2. Is $object already an interceptor?
|
||||
// directly return interceptor objects
|
||||
if ($object instanceof Interceptor) {
|
||||
return $object;
|
||||
}
|
||||
|
||||
// 3. Does an interceptor class for $object exist?
|
||||
// check for an interceptor class
|
||||
$interceptors = $kirby->option('kql.interceptors', []);
|
||||
$interceptors = array_change_key_case($interceptors, CASE_LOWER);
|
||||
// load an interceptor from config if it exists and otherwise fall back to a built-in interceptor
|
||||
$interceptor = $interceptors[$name] ?? static::class($class);
|
||||
|
||||
// check for a valid interceptor class
|
||||
if ($class !== $interceptor && class_exists($interceptor) === true) {
|
||||
return new $interceptor($object);
|
||||
}
|
||||
|
||||
// 4. Also check for parent classes of $object
|
||||
// go through parents of the current object to use their interceptors as fallback
|
||||
foreach (class_parents($object) as $parent) {
|
||||
$interceptor = static::class($parent);
|
||||
|
||||
if (class_exists($interceptor) === true) {
|
||||
return new $interceptor($object);
|
||||
}
|
||||
}
|
||||
|
||||
// 5. $object has no interceptor but is explicitly allowed?
|
||||
// check for a class in the allow list
|
||||
$allowed = $kirby->option('kql.classes.allowed', []);
|
||||
$allowed = array_map('strtolower', $allowed);
|
||||
|
||||
// return the plain object if it is allowed
|
||||
if (in_array($name, $allowed) === true) {
|
||||
return $object;
|
||||
}
|
||||
|
||||
// 6. None of the above? Block class.
|
||||
throw new PermissionException('Access to the class "' . $class . '" is not supported');
|
||||
}
|
||||
|
||||
public function toArray(): array|null
|
||||
{
|
||||
$toArray = [];
|
||||
|
||||
// filter methods which cannot be called
|
||||
foreach ($this->toArray as $method) {
|
||||
if ($this->isAllowedMethod($method) === true) {
|
||||
$toArray[] = $method;
|
||||
}
|
||||
}
|
||||
|
||||
return Kql::select($this, $toArray);
|
||||
}
|
||||
|
||||
/**
|
||||
* Mirrors by default ::toArray but can be
|
||||
* implemented differently by specifc interceptor.
|
||||
* KQL will prefer ::toResponse over ::toArray
|
||||
*/
|
||||
public function toResponse()
|
||||
{
|
||||
return $this->toArray();
|
||||
}
|
||||
}
|
||||
39
public/site/plugins/kql/src/Kql/Interceptors/Cms/App.php
Normal file
39
public/site/plugins/kql/src/Kql/Interceptors/Cms/App.php
Normal file
|
|
@ -0,0 +1,39 @@
|
|||
<?php
|
||||
|
||||
namespace Kirby\Kql\Interceptors\Cms;
|
||||
|
||||
use Kirby\Kql\Interceptor;
|
||||
|
||||
class App extends Interceptor
|
||||
{
|
||||
public const CLASS_ALIAS = 'kirby';
|
||||
|
||||
protected $toArray = [
|
||||
'site',
|
||||
'url'
|
||||
];
|
||||
|
||||
public function allowedMethods(): array
|
||||
{
|
||||
return [
|
||||
'collection',
|
||||
'defaultLanguage',
|
||||
'detectedLanguage',
|
||||
'draft',
|
||||
'file',
|
||||
'language',
|
||||
'languageCode',
|
||||
'languages',
|
||||
'multilang',
|
||||
'page',
|
||||
'roles',
|
||||
'site',
|
||||
'translation',
|
||||
'translations',
|
||||
'url',
|
||||
'user',
|
||||
'users',
|
||||
'version'
|
||||
];
|
||||
}
|
||||
}
|
||||
34
public/site/plugins/kql/src/Kql/Interceptors/Cms/Block.php
Executable file
34
public/site/plugins/kql/src/Kql/Interceptors/Cms/Block.php
Executable file
|
|
@ -0,0 +1,34 @@
|
|||
<?php
|
||||
|
||||
namespace Kirby\Kql\Interceptors\Cms;
|
||||
|
||||
class Block extends Model
|
||||
{
|
||||
public const CLASS_ALIAS = 'block';
|
||||
|
||||
protected $toArray = [
|
||||
'content',
|
||||
'id',
|
||||
'isEmpty',
|
||||
'isHidden',
|
||||
'type'
|
||||
];
|
||||
|
||||
public function allowedMethods(): array
|
||||
{
|
||||
return array_merge(
|
||||
$this->allowedMethodsForSiblings(),
|
||||
[
|
||||
'content',
|
||||
'id',
|
||||
'isEmpty',
|
||||
'isHidden',
|
||||
'isNotEmpty',
|
||||
'toField',
|
||||
'toHtml',
|
||||
'parent',
|
||||
'type'
|
||||
]
|
||||
);
|
||||
}
|
||||
}
|
||||
24
public/site/plugins/kql/src/Kql/Interceptors/Cms/Blocks.php
Executable file
24
public/site/plugins/kql/src/Kql/Interceptors/Cms/Blocks.php
Executable file
|
|
@ -0,0 +1,24 @@
|
|||
<?php
|
||||
|
||||
namespace Kirby\Kql\Interceptors\Cms;
|
||||
|
||||
class Blocks extends Collection
|
||||
{
|
||||
public const CLASS_ALIAS = 'blocks';
|
||||
|
||||
public function allowedMethods(): array
|
||||
{
|
||||
return array_merge(
|
||||
parent::allowedMethods(),
|
||||
[
|
||||
'excerpt',
|
||||
'toHtml'
|
||||
]
|
||||
);
|
||||
}
|
||||
|
||||
public function toArray(): array
|
||||
{
|
||||
return $this->object->toArray();
|
||||
}
|
||||
}
|
||||
|
|
@ -0,0 +1,72 @@
|
|||
<?php
|
||||
|
||||
namespace Kirby\Kql\Interceptors\Cms;
|
||||
|
||||
use Kirby\Kql\Interceptor;
|
||||
|
||||
class Blueprint extends Interceptor
|
||||
{
|
||||
public const CLASS_ALIAS = 'blueprint';
|
||||
|
||||
protected $toArray = [
|
||||
'description',
|
||||
'fields',
|
||||
'isDefault',
|
||||
'name',
|
||||
'sections',
|
||||
'options',
|
||||
'tabs',
|
||||
'title',
|
||||
];
|
||||
|
||||
public function allowedMethods(): array
|
||||
{
|
||||
return [
|
||||
'description',
|
||||
'field',
|
||||
'fields',
|
||||
'isDefault',
|
||||
'name',
|
||||
'options',
|
||||
'section',
|
||||
'sections',
|
||||
'tab',
|
||||
'tabs',
|
||||
'title',
|
||||
];
|
||||
}
|
||||
|
||||
public function fields(): array
|
||||
{
|
||||
return $this->object->fields();
|
||||
}
|
||||
|
||||
public function sections(): array
|
||||
{
|
||||
return array_keys($this->object->sections());
|
||||
}
|
||||
|
||||
public function tab(string $name): ?array
|
||||
{
|
||||
if ($tab = $this->object->tab($name)) {
|
||||
foreach ($tab['columns'] as $columnIndex => $column) {
|
||||
$tab['columns'][$columnIndex]['sections'] = array_keys($column['sections']);
|
||||
}
|
||||
|
||||
return $tab;
|
||||
}
|
||||
|
||||
return null;
|
||||
}
|
||||
|
||||
public function tabs(): array
|
||||
{
|
||||
$tabs = [];
|
||||
|
||||
foreach ($this->object->tabs() as $tab) {
|
||||
$tabs[$tab['name']] = $this->tab($tab['name']);
|
||||
}
|
||||
|
||||
return $tabs;
|
||||
}
|
||||
}
|
||||
|
|
@ -0,0 +1,49 @@
|
|||
<?php
|
||||
|
||||
namespace Kirby\Kql\Interceptors\Cms;
|
||||
|
||||
use Kirby\Kql\Interceptor;
|
||||
|
||||
class Collection extends Interceptor
|
||||
{
|
||||
public const CLASS_ALIAS = 'collection';
|
||||
|
||||
public function allowedMethods(): array
|
||||
{
|
||||
return [
|
||||
'chunk',
|
||||
'count',
|
||||
'filterBy',
|
||||
'find',
|
||||
'findBy',
|
||||
'findByKey',
|
||||
'first',
|
||||
'flip',
|
||||
'groupBy',
|
||||
'has',
|
||||
'isEmpty',
|
||||
'isEven',
|
||||
'isNotEmpty',
|
||||
'isOdd',
|
||||
'keys',
|
||||
'last',
|
||||
'limit',
|
||||
'next',
|
||||
'not',
|
||||
'nth',
|
||||
'offset',
|
||||
'pagination',
|
||||
'pluck',
|
||||
'prev',
|
||||
'shuffle',
|
||||
'slice',
|
||||
'sortBy',
|
||||
'without',
|
||||
];
|
||||
}
|
||||
|
||||
public function toArray(): array
|
||||
{
|
||||
return $this->object->keys();
|
||||
}
|
||||
}
|
||||
71
public/site/plugins/kql/src/Kql/Interceptors/Cms/File.php
Normal file
71
public/site/plugins/kql/src/Kql/Interceptors/Cms/File.php
Normal file
|
|
@ -0,0 +1,71 @@
|
|||
<?php
|
||||
|
||||
namespace Kirby\Kql\Interceptors\Cms;
|
||||
|
||||
class File extends Model
|
||||
{
|
||||
public const CLASS_ALIAS = 'file';
|
||||
|
||||
protected $toArray = [
|
||||
'extension',
|
||||
'filename',
|
||||
'height',
|
||||
'id',
|
||||
'mime',
|
||||
'niceSize',
|
||||
'template',
|
||||
'type',
|
||||
'url',
|
||||
'width'
|
||||
];
|
||||
|
||||
public function allowedMethods(): array
|
||||
{
|
||||
return array_merge(
|
||||
$this->allowedMethodsForModels(),
|
||||
$this->allowedMethodsForParents(),
|
||||
$this->allowedMethodsForSiblings(),
|
||||
[
|
||||
'blur',
|
||||
'bw',
|
||||
'crop',
|
||||
'dataUri',
|
||||
'dimensions',
|
||||
'exif',
|
||||
'extension',
|
||||
'filename',
|
||||
'files',
|
||||
'grayscale',
|
||||
'greyscale',
|
||||
'height',
|
||||
'html',
|
||||
'isPortrait',
|
||||
'isLandscape',
|
||||
'isSquare',
|
||||
'mime',
|
||||
'name',
|
||||
'niceSize',
|
||||
'orientation',
|
||||
'ratio',
|
||||
'resize',
|
||||
'size',
|
||||
'srcset',
|
||||
'template',
|
||||
'templateSiblings',
|
||||
'thumb',
|
||||
'type',
|
||||
'width'
|
||||
]
|
||||
);
|
||||
}
|
||||
|
||||
public function dimensions(): array
|
||||
{
|
||||
return $this->object->dimensions()->toArray();
|
||||
}
|
||||
|
||||
public function exif(): array
|
||||
{
|
||||
return $this->object->exif()->toArray();
|
||||
}
|
||||
}
|
||||
|
|
@ -0,0 +1,8 @@
|
|||
<?php
|
||||
|
||||
namespace Kirby\Kql\Interceptors\Cms;
|
||||
|
||||
class FileVersion extends File
|
||||
{
|
||||
public const CLASS_ALIAS = 'file';
|
||||
}
|
||||
18
public/site/plugins/kql/src/Kql/Interceptors/Cms/Files.php
Normal file
18
public/site/plugins/kql/src/Kql/Interceptors/Cms/Files.php
Normal file
|
|
@ -0,0 +1,18 @@
|
|||
<?php
|
||||
|
||||
namespace Kirby\Kql\Interceptors\Cms;
|
||||
|
||||
class Files extends Collection
|
||||
{
|
||||
public const CLASS_ALIAS = 'files';
|
||||
|
||||
public function allowedMethods(): array
|
||||
{
|
||||
return array_merge(
|
||||
parent::allowedMethods(),
|
||||
[
|
||||
'template'
|
||||
]
|
||||
);
|
||||
}
|
||||
}
|
||||
30
public/site/plugins/kql/src/Kql/Interceptors/Cms/Layout.php
Executable file
30
public/site/plugins/kql/src/Kql/Interceptors/Cms/Layout.php
Executable file
|
|
@ -0,0 +1,30 @@
|
|||
<?php
|
||||
|
||||
namespace Kirby\Kql\Interceptors\Cms;
|
||||
|
||||
class Layout extends Model
|
||||
{
|
||||
public const CLASS_ALIAS = 'layout';
|
||||
|
||||
protected $toArray = [
|
||||
'attrs',
|
||||
'columns',
|
||||
'id',
|
||||
'isEmpty',
|
||||
];
|
||||
|
||||
public function allowedMethods(): array
|
||||
{
|
||||
return array_merge(
|
||||
$this->allowedMethodsForSiblings(),
|
||||
[
|
||||
'attrs',
|
||||
'columns',
|
||||
'id',
|
||||
'isEmpty',
|
||||
'isNotEmpty',
|
||||
'parent'
|
||||
]
|
||||
);
|
||||
}
|
||||
}
|
||||
30
public/site/plugins/kql/src/Kql/Interceptors/Cms/LayoutColumn.php
Executable file
30
public/site/plugins/kql/src/Kql/Interceptors/Cms/LayoutColumn.php
Executable file
|
|
@ -0,0 +1,30 @@
|
|||
<?php
|
||||
|
||||
namespace Kirby\Kql\Interceptors\Cms;
|
||||
|
||||
class LayoutColumn extends Model
|
||||
{
|
||||
public const CLASS_ALIAS = 'layoutColumn';
|
||||
|
||||
protected $toArray = [
|
||||
'blocks',
|
||||
'id',
|
||||
'isEmpty',
|
||||
'width',
|
||||
];
|
||||
|
||||
public function allowedMethods(): array
|
||||
{
|
||||
return array_merge(
|
||||
$this->allowedMethodsForSiblings(),
|
||||
[
|
||||
'blocks',
|
||||
'id',
|
||||
'isEmpty',
|
||||
'isNotEmpty',
|
||||
'span',
|
||||
'width'
|
||||
]
|
||||
);
|
||||
}
|
||||
}
|
||||
13
public/site/plugins/kql/src/Kql/Interceptors/Cms/LayoutColumns.php
Executable file
13
public/site/plugins/kql/src/Kql/Interceptors/Cms/LayoutColumns.php
Executable file
|
|
@ -0,0 +1,13 @@
|
|||
<?php
|
||||
|
||||
namespace Kirby\Kql\Interceptors\Cms;
|
||||
|
||||
class LayoutColumns extends Collection
|
||||
{
|
||||
public const CLASS_ALIAS = 'layoutColumns';
|
||||
|
||||
public function toArray(): array
|
||||
{
|
||||
return $this->object->toArray();
|
||||
}
|
||||
}
|
||||
13
public/site/plugins/kql/src/Kql/Interceptors/Cms/Layouts.php
Executable file
13
public/site/plugins/kql/src/Kql/Interceptors/Cms/Layouts.php
Executable file
|
|
@ -0,0 +1,13 @@
|
|||
<?php
|
||||
|
||||
namespace Kirby\Kql\Interceptors\Cms;
|
||||
|
||||
class Layouts extends Collection
|
||||
{
|
||||
public const CLASS_ALIAS = 'layouts';
|
||||
|
||||
public function toArray(): array
|
||||
{
|
||||
return $this->object->toArray();
|
||||
}
|
||||
}
|
||||
113
public/site/plugins/kql/src/Kql/Interceptors/Cms/Model.php
Normal file
113
public/site/plugins/kql/src/Kql/Interceptors/Cms/Model.php
Normal file
|
|
@ -0,0 +1,113 @@
|
|||
<?php
|
||||
|
||||
namespace Kirby\Kql\Interceptors\Cms;
|
||||
|
||||
use Kirby\Kql\Interceptor;
|
||||
|
||||
class Model extends Interceptor
|
||||
{
|
||||
public const CLASS_ALIAS = 'model';
|
||||
|
||||
public function __call($method, array $args = [])
|
||||
{
|
||||
if ($this->isAllowedMethod($method) === true) {
|
||||
return $this->object->$method(...$args);
|
||||
}
|
||||
|
||||
if (method_exists($this->object, $method) === false) {
|
||||
return $this->object->content()->get($method);
|
||||
}
|
||||
|
||||
$this->forbiddenMethod($method);
|
||||
}
|
||||
|
||||
protected function allowedMethodsForChildren()
|
||||
{
|
||||
return [
|
||||
'children',
|
||||
'childrenAndDrafts',
|
||||
'draft',
|
||||
'drafts',
|
||||
'find',
|
||||
'findPageOrDraft',
|
||||
'grandChildren',
|
||||
'hasChildren',
|
||||
'hasDrafts',
|
||||
'hasListedChildren',
|
||||
'hasUnlistedChildren',
|
||||
'index',
|
||||
'search',
|
||||
];
|
||||
}
|
||||
|
||||
protected function allowedMethodsForFiles()
|
||||
{
|
||||
return [
|
||||
'audio',
|
||||
'code',
|
||||
'documents',
|
||||
'file',
|
||||
'files',
|
||||
'hasAudio',
|
||||
'hasCode',
|
||||
'hasDocuments',
|
||||
'hasFiles',
|
||||
'hasImages',
|
||||
'hasVideos',
|
||||
'image',
|
||||
'images',
|
||||
'videos'
|
||||
];
|
||||
}
|
||||
|
||||
protected function allowedMethodsForModels()
|
||||
{
|
||||
return [
|
||||
'apiUrl',
|
||||
'blueprint',
|
||||
'content',
|
||||
'dragText',
|
||||
'exists',
|
||||
'id',
|
||||
'mediaUrl',
|
||||
'modified',
|
||||
'permissions',
|
||||
'panel',
|
||||
'permalink',
|
||||
'previewUrl',
|
||||
'url',
|
||||
];
|
||||
}
|
||||
|
||||
protected function allowedMethodsForSiblings()
|
||||
{
|
||||
return [
|
||||
'indexOf',
|
||||
'next',
|
||||
'nextAll',
|
||||
'prev',
|
||||
'prevAll',
|
||||
'siblings',
|
||||
'hasNext',
|
||||
'hasPrev',
|
||||
'isFirst',
|
||||
'isLast',
|
||||
'isNth'
|
||||
];
|
||||
}
|
||||
|
||||
protected function allowedMethodsForParents()
|
||||
{
|
||||
return [
|
||||
'parent',
|
||||
'parentId',
|
||||
'parentModel',
|
||||
'site',
|
||||
];
|
||||
}
|
||||
|
||||
public function uuid(): string
|
||||
{
|
||||
return $this->object->uuid()->toString();
|
||||
}
|
||||
}
|
||||
68
public/site/plugins/kql/src/Kql/Interceptors/Cms/Page.php
Normal file
68
public/site/plugins/kql/src/Kql/Interceptors/Cms/Page.php
Normal file
|
|
@ -0,0 +1,68 @@
|
|||
<?php
|
||||
|
||||
namespace Kirby\Kql\Interceptors\Cms;
|
||||
|
||||
class Page extends Model
|
||||
{
|
||||
public const CLASS_ALIAS = 'page';
|
||||
|
||||
protected $toArray = [
|
||||
'children',
|
||||
'content',
|
||||
'drafts',
|
||||
'files',
|
||||
'id',
|
||||
'intendedTemplate',
|
||||
'isHomePage',
|
||||
'isErrorPage',
|
||||
'num',
|
||||
'template',
|
||||
'title',
|
||||
'slug',
|
||||
'status',
|
||||
'uid',
|
||||
'url'
|
||||
];
|
||||
|
||||
public function allowedMethods(): array
|
||||
{
|
||||
return array_merge(
|
||||
$this->allowedMethodsForChildren(),
|
||||
$this->allowedMethodsForFiles(),
|
||||
$this->allowedMethodsForModels(),
|
||||
$this->allowedMethodsForParents(),
|
||||
$this->allowedMethodsForSiblings(),
|
||||
[
|
||||
'blueprints',
|
||||
'depth',
|
||||
'hasTemplate',
|
||||
'intendedTemplate',
|
||||
'isDraft',
|
||||
'isErrorPage',
|
||||
'isHomePage',
|
||||
'isHomeOrErrorPage',
|
||||
'isListed',
|
||||
'isReadable',
|
||||
'isSortable',
|
||||
'isUnlisted',
|
||||
'num',
|
||||
'slug',
|
||||
'status',
|
||||
'template',
|
||||
'title',
|
||||
'uid',
|
||||
'uri',
|
||||
]
|
||||
);
|
||||
}
|
||||
|
||||
public function intendedTemplate(): string
|
||||
{
|
||||
return $this->object->intendedTemplate()->name();
|
||||
}
|
||||
|
||||
public function template(): string
|
||||
{
|
||||
return $this->object->template()->name();
|
||||
}
|
||||
}
|
||||
34
public/site/plugins/kql/src/Kql/Interceptors/Cms/Pages.php
Normal file
34
public/site/plugins/kql/src/Kql/Interceptors/Cms/Pages.php
Normal file
|
|
@ -0,0 +1,34 @@
|
|||
<?php
|
||||
|
||||
namespace Kirby\Kql\Interceptors\Cms;
|
||||
|
||||
class Pages extends Collection
|
||||
{
|
||||
public const CLASS_ALIAS = 'pages';
|
||||
|
||||
public function allowedMethods(): array
|
||||
{
|
||||
return array_merge(
|
||||
parent::allowedMethods(),
|
||||
[
|
||||
'audio',
|
||||
'children',
|
||||
'code',
|
||||
'documents',
|
||||
'drafts',
|
||||
'files',
|
||||
'findByUri',
|
||||
'images',
|
||||
'index',
|
||||
'listed',
|
||||
'notTemplate',
|
||||
'nums',
|
||||
'published',
|
||||
'search',
|
||||
'template',
|
||||
'unlisted',
|
||||
'videos',
|
||||
]
|
||||
);
|
||||
}
|
||||
}
|
||||
33
public/site/plugins/kql/src/Kql/Interceptors/Cms/Role.php
Normal file
33
public/site/plugins/kql/src/Kql/Interceptors/Cms/Role.php
Normal file
|
|
@ -0,0 +1,33 @@
|
|||
<?php
|
||||
|
||||
namespace Kirby\Kql\Interceptors\Cms;
|
||||
|
||||
use Kirby\Kql\Interceptor;
|
||||
|
||||
class Role extends Interceptor
|
||||
{
|
||||
public const CLASS_ALIAS = 'role';
|
||||
|
||||
protected $toArray = [
|
||||
'description',
|
||||
'id',
|
||||
'name',
|
||||
'title',
|
||||
];
|
||||
|
||||
public function allowedMethods(): array
|
||||
{
|
||||
return [
|
||||
'description',
|
||||
'id',
|
||||
'name',
|
||||
'permissions',
|
||||
'title'
|
||||
];
|
||||
}
|
||||
|
||||
public function permissions(): array
|
||||
{
|
||||
return $this->object->permissions()->toArray();
|
||||
}
|
||||
}
|
||||
36
public/site/plugins/kql/src/Kql/Interceptors/Cms/Site.php
Normal file
36
public/site/plugins/kql/src/Kql/Interceptors/Cms/Site.php
Normal file
|
|
@ -0,0 +1,36 @@
|
|||
<?php
|
||||
|
||||
namespace Kirby\Kql\Interceptors\Cms;
|
||||
|
||||
class Site extends Model
|
||||
{
|
||||
public const CLASS_ALIAS = 'site';
|
||||
|
||||
protected $toArray = [
|
||||
'children',
|
||||
'drafts',
|
||||
'files',
|
||||
'title',
|
||||
'url',
|
||||
];
|
||||
|
||||
public function allowedMethods(): array
|
||||
{
|
||||
return array_merge(
|
||||
$this->allowedMethodsForChildren(),
|
||||
$this->allowedMethodsForFiles(),
|
||||
$this->allowedMethodsForModels(),
|
||||
[
|
||||
'blueprints',
|
||||
'breadcrumb',
|
||||
'errorPage',
|
||||
'errorPageId',
|
||||
'homePage',
|
||||
'homePageId',
|
||||
'page',
|
||||
'pages',
|
||||
'title',
|
||||
]
|
||||
);
|
||||
}
|
||||
}
|
||||
|
|
@ -0,0 +1,13 @@
|
|||
<?php
|
||||
|
||||
namespace Kirby\Kql\Interceptors\Cms;
|
||||
|
||||
class Structure extends Collection
|
||||
{
|
||||
public const CLASS_ALIAS = 'structure';
|
||||
|
||||
public function toArray(): array
|
||||
{
|
||||
return $this->object->toArray();
|
||||
}
|
||||
}
|
||||
|
|
@ -0,0 +1,20 @@
|
|||
<?php
|
||||
|
||||
namespace Kirby\Kql\Interceptors\Cms;
|
||||
|
||||
class StructureObject extends Model
|
||||
{
|
||||
public const CLASS_ALIAS = 'structureItem';
|
||||
|
||||
public function allowedMethods(): array
|
||||
{
|
||||
return array_merge(
|
||||
$this->allowedMethodsForSiblings(),
|
||||
[
|
||||
'content',
|
||||
'id',
|
||||
'parent',
|
||||
]
|
||||
);
|
||||
}
|
||||
}
|
||||
35
public/site/plugins/kql/src/Kql/Interceptors/Cms/Translation.php
Executable file
35
public/site/plugins/kql/src/Kql/Interceptors/Cms/Translation.php
Executable file
|
|
@ -0,0 +1,35 @@
|
|||
<?php
|
||||
|
||||
namespace Kirby\Kql\Interceptors\Cms;
|
||||
|
||||
use Kirby\Kql\Interceptor;
|
||||
|
||||
class Translation extends Interceptor
|
||||
{
|
||||
public const CLASS_ALIAS = 'translation';
|
||||
|
||||
protected $toArray = [
|
||||
'code',
|
||||
'data',
|
||||
'direction',
|
||||
'id',
|
||||
'name',
|
||||
'locale',
|
||||
'author'
|
||||
];
|
||||
|
||||
public function allowedMethods(): array
|
||||
{
|
||||
return [
|
||||
'code',
|
||||
'data',
|
||||
'dataWithFallback',
|
||||
'direction',
|
||||
'get',
|
||||
'id',
|
||||
'name',
|
||||
'locale',
|
||||
'author'
|
||||
];
|
||||
}
|
||||
}
|
||||
35
public/site/plugins/kql/src/Kql/Interceptors/Cms/User.php
Normal file
35
public/site/plugins/kql/src/Kql/Interceptors/Cms/User.php
Normal file
|
|
@ -0,0 +1,35 @@
|
|||
<?php
|
||||
|
||||
namespace Kirby\Kql\Interceptors\Cms;
|
||||
|
||||
class User extends Model
|
||||
{
|
||||
public const CLASS_ALIAS = 'user';
|
||||
|
||||
protected $toArray = [
|
||||
'id',
|
||||
'name',
|
||||
'role',
|
||||
'username'
|
||||
];
|
||||
|
||||
public function allowedMethods(): array
|
||||
{
|
||||
return array_merge(
|
||||
$this->allowedMethodsForFiles(),
|
||||
$this->allowedMethodsForModels(),
|
||||
$this->allowedMethodsForSiblings(),
|
||||
[
|
||||
'avatar',
|
||||
'email',
|
||||
'id',
|
||||
'isAdmin',
|
||||
'language',
|
||||
'modified',
|
||||
'name',
|
||||
'role',
|
||||
'username',
|
||||
]
|
||||
);
|
||||
}
|
||||
}
|
||||
18
public/site/plugins/kql/src/Kql/Interceptors/Cms/Users.php
Normal file
18
public/site/plugins/kql/src/Kql/Interceptors/Cms/Users.php
Normal file
|
|
@ -0,0 +1,18 @@
|
|||
<?php
|
||||
|
||||
namespace Kirby\Kql\Interceptors\Cms;
|
||||
|
||||
class Users extends Collection
|
||||
{
|
||||
public const CLASS_ALIAS = 'users';
|
||||
|
||||
public function allowedMethods(): array
|
||||
{
|
||||
return array_merge(
|
||||
parent::allowedMethods(),
|
||||
[
|
||||
'role'
|
||||
]
|
||||
);
|
||||
}
|
||||
}
|
||||
|
|
@ -0,0 +1,40 @@
|
|||
<?php
|
||||
|
||||
namespace Kirby\Kql\Interceptors\Content;
|
||||
|
||||
use Kirby\Kql\Interceptor;
|
||||
|
||||
class Content extends Interceptor
|
||||
{
|
||||
public const CLASS_ALIAS = 'content';
|
||||
|
||||
public function __call($method, array $args = [])
|
||||
{
|
||||
if ($this->isAllowedMethod($method) === true) {
|
||||
return $this->object->$method(...$args);
|
||||
}
|
||||
|
||||
if (method_exists($this->object, $method) === false) {
|
||||
return $this->object->get($method);
|
||||
}
|
||||
|
||||
$this->forbiddenMethod($method);
|
||||
}
|
||||
|
||||
public function allowedMethods(): array
|
||||
{
|
||||
return [
|
||||
'data',
|
||||
'fields',
|
||||
'has',
|
||||
'get',
|
||||
'keys',
|
||||
'not',
|
||||
];
|
||||
}
|
||||
|
||||
public function toArray(): array
|
||||
{
|
||||
return $this->object->toArray();
|
||||
}
|
||||
}
|
||||
|
|
@ -0,0 +1,52 @@
|
|||
<?php
|
||||
|
||||
namespace Kirby\Kql\Interceptors\Content;
|
||||
|
||||
use Kirby\Kql\Interceptor;
|
||||
|
||||
class Field extends Interceptor
|
||||
{
|
||||
public const CLASS_ALIAS = 'field';
|
||||
|
||||
public function __call($method, array $args = [])
|
||||
{
|
||||
if ($this->isAllowedMethod($method) === true) {
|
||||
return $this->object->$method(...$args);
|
||||
}
|
||||
|
||||
// field methods
|
||||
$methods = array_keys($this->object::$methods);
|
||||
$method = strtolower($method);
|
||||
|
||||
if (in_array($method, $methods) === true) {
|
||||
return $this->object->$method(...$args);
|
||||
}
|
||||
|
||||
// aliases
|
||||
$aliases = array_keys($this->object::$aliases);
|
||||
$alias = strtolower($method);
|
||||
|
||||
if (in_array($alias, $aliases) === true) {
|
||||
return $this->object->$method(...$args);
|
||||
}
|
||||
|
||||
$this->forbiddenMethod($method);
|
||||
}
|
||||
|
||||
public function allowedMethods(): array
|
||||
{
|
||||
return [
|
||||
'exists',
|
||||
'isEmpty',
|
||||
'isNotEmpty',
|
||||
'key',
|
||||
'or',
|
||||
'value'
|
||||
];
|
||||
}
|
||||
|
||||
public function toResponse()
|
||||
{
|
||||
return $this->object->toString();
|
||||
}
|
||||
}
|
||||
30
public/site/plugins/kql/src/Kql/Interceptors/Panel/Model.php
Executable file
30
public/site/plugins/kql/src/Kql/Interceptors/Panel/Model.php
Executable file
|
|
@ -0,0 +1,30 @@
|
|||
<?php
|
||||
|
||||
namespace Kirby\Kql\Interceptors\Panel;
|
||||
|
||||
use Kirby\Kql\Interceptor;
|
||||
|
||||
class Model extends Interceptor
|
||||
{
|
||||
public const CLASS_ALIAS = 'panel';
|
||||
|
||||
public function allowedMethods(): array
|
||||
{
|
||||
return [
|
||||
'dragText',
|
||||
'image',
|
||||
'path',
|
||||
'url',
|
||||
];
|
||||
}
|
||||
|
||||
public function toArray(): array
|
||||
{
|
||||
return [
|
||||
'dragText' => $this->dragText(),
|
||||
'image' => $this->image(),
|
||||
'path' => $this->path(),
|
||||
'url' => $this->url(),
|
||||
];
|
||||
}
|
||||
}
|
||||
24
public/site/plugins/kql/src/Kql/Interceptors/Toolkit/Obj.php
Executable file
24
public/site/plugins/kql/src/Kql/Interceptors/Toolkit/Obj.php
Executable file
|
|
@ -0,0 +1,24 @@
|
|||
<?php
|
||||
|
||||
namespace Kirby\Kql\Interceptors\Toolkit;
|
||||
|
||||
use Kirby\Kql\Interceptor;
|
||||
|
||||
class Obj extends Interceptor
|
||||
{
|
||||
public const CLASS_ALIAS = 'obj';
|
||||
|
||||
public function allowedMethods(): array
|
||||
{
|
||||
return [
|
||||
'get',
|
||||
'toArray',
|
||||
'toJson',
|
||||
];
|
||||
}
|
||||
|
||||
public function toArray(): array
|
||||
{
|
||||
return $this->object->toArray();
|
||||
}
|
||||
}
|
||||
226
public/site/plugins/kql/src/Kql/Kql.php
Normal file
226
public/site/plugins/kql/src/Kql/Kql.php
Normal file
|
|
@ -0,0 +1,226 @@
|
|||
<?php
|
||||
|
||||
namespace Kirby\Kql;
|
||||
|
||||
use Exception;
|
||||
use Kirby\Cms\App;
|
||||
use Kirby\Cms\Collection;
|
||||
use Kirby\Toolkit\Str;
|
||||
|
||||
/**
|
||||
* ...
|
||||
*
|
||||
* @package Kirby KQL
|
||||
* @author Bastian Allgeier <bastian@getkirby.com>
|
||||
* @link https://getkirby.com
|
||||
* @copyright Bastian Allgeier
|
||||
* @license https://getkirby.com/license
|
||||
*/
|
||||
class Kql
|
||||
{
|
||||
public static function fetch($model, $key, $selection)
|
||||
{
|
||||
// simple key/value
|
||||
if ($selection === true) {
|
||||
return static::render($model->$key());
|
||||
}
|
||||
|
||||
// selection without additional query
|
||||
if (
|
||||
is_array($selection) === true &&
|
||||
empty($selection['query']) === true
|
||||
) {
|
||||
return static::select(
|
||||
$model->$key(),
|
||||
$selection['select'] ?? null,
|
||||
$selection['options'] ?? []
|
||||
);
|
||||
}
|
||||
|
||||
// nested queries
|
||||
return static::run($selection, $model);
|
||||
}
|
||||
|
||||
/**
|
||||
* Returns helpful information about the object
|
||||
* type as well as, if available, values and methods
|
||||
*/
|
||||
public static function help($object): array
|
||||
{
|
||||
return Help::for($object);
|
||||
}
|
||||
|
||||
public static function query(string $query, $model = null)
|
||||
{
|
||||
$model ??= App::instance()->site();
|
||||
$data = [$model::CLASS_ALIAS => $model];
|
||||
|
||||
return Query::factory($query)->resolve($data);
|
||||
}
|
||||
|
||||
public static function render($value)
|
||||
{
|
||||
if (is_object($value) === true) {
|
||||
// replace actual object with intercepting proxy class
|
||||
$object = Interceptor::replace($value);
|
||||
|
||||
if (method_exists($object, 'toResponse') === true) {
|
||||
return $object->toResponse();
|
||||
}
|
||||
|
||||
if (method_exists($object, 'toArray') === true) {
|
||||
return $object->toArray();
|
||||
}
|
||||
|
||||
throw new Exception('The object "' . get_class($object) . '" cannot be rendered. Try querying one of its methods instead.');
|
||||
}
|
||||
|
||||
return $value;
|
||||
}
|
||||
|
||||
public static function run($input, $model = null)
|
||||
{
|
||||
// string queries
|
||||
if (is_string($input) === true) {
|
||||
$result = static::query($input, $model);
|
||||
return static::render($result);
|
||||
}
|
||||
|
||||
// multiple queries
|
||||
if (isset($input['queries']) === true) {
|
||||
$result = [];
|
||||
|
||||
foreach ($input['queries'] as $name => $query) {
|
||||
$result[$name] = static::run($query);
|
||||
}
|
||||
|
||||
return $result;
|
||||
}
|
||||
|
||||
$query = $input['query'] ?? 'site';
|
||||
$select = $input['select'] ?? null;
|
||||
$options = ['pagination' => $input['pagination'] ?? null];
|
||||
|
||||
// check for invalid queries
|
||||
if (is_string($query) === false) {
|
||||
throw new Exception('The query must be a string');
|
||||
}
|
||||
|
||||
$result = static::query($query, $model);
|
||||
return static::select($result, $select, $options);
|
||||
}
|
||||
|
||||
public static function select(
|
||||
$data,
|
||||
array|string|null $select = null,
|
||||
array $options = []
|
||||
) {
|
||||
if ($select === null) {
|
||||
return static::render($data);
|
||||
}
|
||||
|
||||
if ($select === '?') {
|
||||
return static::help($data);
|
||||
}
|
||||
|
||||
if ($data instanceof Collection) {
|
||||
return static::selectFromCollection($data, $select, $options);
|
||||
}
|
||||
|
||||
if (is_object($data) === true) {
|
||||
return static::selectFromObject($data, $select);
|
||||
}
|
||||
|
||||
if (is_array($data) === true) {
|
||||
return static::selectFromArray($data, $select);
|
||||
}
|
||||
}
|
||||
|
||||
/**
|
||||
* @internal
|
||||
*/
|
||||
public static function selectFromArray(array $array, array $select): array
|
||||
{
|
||||
$result = [];
|
||||
|
||||
foreach ($select as $key => $selection) {
|
||||
if ($selection === false) {
|
||||
continue;
|
||||
}
|
||||
|
||||
if (is_int($key) === true) {
|
||||
$key = $selection;
|
||||
$selection = true;
|
||||
}
|
||||
|
||||
$result[$key] = $array[$key] ?? null;
|
||||
}
|
||||
|
||||
return $result;
|
||||
}
|
||||
|
||||
/**
|
||||
* @internal
|
||||
*/
|
||||
public static function selectFromCollection(
|
||||
Collection $collection,
|
||||
array|string $select,
|
||||
array $options = []
|
||||
): array {
|
||||
if ($options['pagination'] ?? false) {
|
||||
$collection = $collection->paginate($options['pagination']);
|
||||
}
|
||||
|
||||
$data = [];
|
||||
|
||||
foreach ($collection as $model) {
|
||||
$data[] = static::selectFromObject($model, $select);
|
||||
}
|
||||
|
||||
if ($pagination = $collection->pagination()) {
|
||||
return [
|
||||
'data' => $data,
|
||||
'pagination' => [
|
||||
'page' => $pagination->page(),
|
||||
'pages' => $pagination->pages(),
|
||||
'offset' => $pagination->offset(),
|
||||
'limit' => $pagination->limit(),
|
||||
'total' => $pagination->total(),
|
||||
],
|
||||
];
|
||||
}
|
||||
|
||||
return $data;
|
||||
}
|
||||
|
||||
/**
|
||||
* @internal
|
||||
*/
|
||||
public static function selectFromObject(
|
||||
object $object,
|
||||
array|string $select
|
||||
): array {
|
||||
// replace actual object with intercepting proxy class
|
||||
$object = Interceptor::replace($object);
|
||||
$result = [];
|
||||
|
||||
if (is_string($select) === true) {
|
||||
$select = Str::split($select);
|
||||
}
|
||||
|
||||
foreach ($select as $key => $selection) {
|
||||
if ($selection === false) {
|
||||
continue;
|
||||
}
|
||||
|
||||
if (is_int($key) === true) {
|
||||
$key = $selection;
|
||||
$selection = true;
|
||||
}
|
||||
|
||||
$result[$key] = static::fetch($object, $key, $selection);
|
||||
}
|
||||
|
||||
return $result;
|
||||
}
|
||||
}
|
||||
29
public/site/plugins/kql/src/Kql/Query.php
Normal file
29
public/site/plugins/kql/src/Kql/Query.php
Normal file
|
|
@ -0,0 +1,29 @@
|
|||
<?php
|
||||
|
||||
namespace Kirby\Kql;
|
||||
|
||||
use Kirby\Query\Query as BaseQuery;
|
||||
|
||||
/**
|
||||
* Extends the core Query class with the KQL-specific
|
||||
* functionalities to intercept the segments chain calls
|
||||
*
|
||||
* @package Kirby KQL
|
||||
* @author Nico Hoffmann <nico@getkirby.com>
|
||||
* @link https://getkirby.com
|
||||
* @copyright Bastian Allgeier
|
||||
* @license https://getkirby.com/license
|
||||
*/
|
||||
class Query extends BaseQuery
|
||||
{
|
||||
/**
|
||||
* Intercepts the chain of segments called
|
||||
* on each other by replacing objects with
|
||||
* their corresponding Interceptor which
|
||||
* handles blocking calls to restricted methods
|
||||
*/
|
||||
public function intercept(mixed $result): mixed
|
||||
{
|
||||
return is_object($result) ? Interceptor::replace($result): $result;
|
||||
}
|
||||
}
|
||||
Loading…
Add table
Add a link
Reference in a new issue