2023-02-14 21:38:42 +11:00
|
|
|
<?php
|
|
|
|
|
|
|
|
namespace App\Classes\LDAP\Schema;
|
|
|
|
|
|
|
|
use Illuminate\Support\Collection;
|
|
|
|
use Illuminate\Support\Facades\Log;
|
|
|
|
|
|
|
|
use App\Classes\LDAP\Server;
|
|
|
|
use App\Exceptions\InvalidUsage;
|
|
|
|
use App\Ldap\Entry;
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Represents an LDAP Schema objectClass
|
|
|
|
*
|
|
|
|
* @package phpLDAPadmin
|
|
|
|
* @subpackage Schema
|
|
|
|
*/
|
2023-04-02 22:08:43 +10:00
|
|
|
final class ObjectClass extends Base {
|
2023-02-14 21:38:42 +11:00
|
|
|
// The server ID that this objectclass belongs to.
|
|
|
|
private Server $server;
|
|
|
|
|
|
|
|
// Array of objectClass names from which this objectClass inherits
|
|
|
|
private Collection $sup_classes;
|
|
|
|
|
|
|
|
// One of STRUCTURAL, ABSTRACT, or AUXILIARY
|
|
|
|
private int $type;
|
|
|
|
|
|
|
|
// Arrays of attribute names that this objectClass requires
|
|
|
|
private Collection $must_attrs;
|
|
|
|
|
|
|
|
// Arrays of attribute names that this objectClass allows, but does not require
|
|
|
|
private Collection $may_attrs;
|
|
|
|
|
|
|
|
// Arrays of attribute names that this objectClass has been forced to MAY attrs, due to configuration
|
|
|
|
private Collection $may_force;
|
|
|
|
|
|
|
|
// Array of objectClasses which inherit from this one
|
|
|
|
private Collection $child_objectclasses;
|
|
|
|
|
|
|
|
private bool $is_obsolete;
|
|
|
|
|
|
|
|
/* ObjectClass Types */
|
|
|
|
private const OC_STRUCTURAL = 0x01;
|
|
|
|
private const OC_ABSTRACT = 0x02;
|
|
|
|
private const OC_AUXILIARY = 0x03;
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Creates a new ObjectClass object given a raw LDAP objectClass string.
|
|
|
|
*
|
|
|
|
* eg: ( 2.5.6.0 NAME 'top' DESC 'top of the superclass chain' ABSTRACT MUST objectClass )
|
|
|
|
*/
|
2023-02-19 00:32:46 +11:00
|
|
|
public function __construct(string $line,Server $server)
|
2023-02-14 21:38:42 +11:00
|
|
|
{
|
|
|
|
parent::__construct($line);
|
|
|
|
|
2024-12-30 22:41:28 +11:00
|
|
|
if (static::DEBUG_VERBOSE)
|
|
|
|
Log::debug(sprintf('Parsing ObjectClass [%s]',$line));
|
2023-02-14 21:38:42 +11:00
|
|
|
|
|
|
|
$strings = preg_split('/[\s,]+/',$line,-1,PREG_SPLIT_DELIM_CAPTURE);
|
|
|
|
|
|
|
|
// Init
|
|
|
|
$this->server = $server;
|
|
|
|
$this->may_attrs = collect();
|
|
|
|
$this->may_force = collect();
|
|
|
|
$this->must_attrs = collect();
|
|
|
|
$this->sup_classes = collect();
|
|
|
|
$this->child_objectclasses = collect();
|
|
|
|
|
|
|
|
for ($i=0; $i < count($strings); $i++) {
|
|
|
|
switch ($strings[$i]) {
|
|
|
|
case '(':
|
|
|
|
case ')':
|
|
|
|
break;
|
|
|
|
|
|
|
|
case 'NAME':
|
|
|
|
if ($strings[$i+1] != '(') {
|
|
|
|
do {
|
|
|
|
$this->name .= (strlen($this->name) ? ' ' : '').$strings[++$i];
|
|
|
|
|
|
|
|
} while (! preg_match('/\'$/s',$strings[$i]));
|
|
|
|
|
|
|
|
} else {
|
|
|
|
$i++;
|
|
|
|
|
|
|
|
do {
|
|
|
|
$this->name .= (strlen($this->name) ? ' ' : '').$strings[++$i];
|
|
|
|
|
|
|
|
} while (! preg_match('/\'$/s',$strings[$i]));
|
|
|
|
|
|
|
|
do {
|
|
|
|
$i++;
|
|
|
|
} while (! preg_match('/\)+\)?/',$strings[$i]));
|
|
|
|
}
|
|
|
|
|
|
|
|
$this->name = preg_replace("/^\'(.*)\'$/",'$1',$this->name);
|
|
|
|
|
2024-12-30 22:41:28 +11:00
|
|
|
if (static::DEBUG_VERBOSE)
|
|
|
|
Log::debug(sprintf(sprintf('- Case NAME returned (%s)',$this->name)));
|
2023-02-14 21:38:42 +11:00
|
|
|
break;
|
|
|
|
|
|
|
|
case 'DESC':
|
|
|
|
do {
|
|
|
|
$this->description .= (strlen($this->description) ? ' ' : '').$strings[++$i];
|
|
|
|
|
|
|
|
} while (! preg_match('/\'$/s',$strings[$i]));
|
|
|
|
|
|
|
|
$this->description = preg_replace("/^\'(.*)\'$/",'$1',$this->description);
|
|
|
|
|
2024-12-30 22:41:28 +11:00
|
|
|
if (static::DEBUG_VERBOSE)
|
|
|
|
Log::debug(sprintf('- Case DESC returned (%s)',$this->description));
|
2023-02-14 21:38:42 +11:00
|
|
|
break;
|
|
|
|
|
|
|
|
case 'OBSOLETE':
|
|
|
|
$this->is_obsolete = TRUE;
|
|
|
|
|
2024-12-30 22:41:28 +11:00
|
|
|
if (static::DEBUG_VERBOSE)
|
|
|
|
Log::debug(sprintf('- Case OBSOLETE returned (%s)',$this->is_obsolete));
|
2023-02-14 21:38:42 +11:00
|
|
|
break;
|
|
|
|
|
|
|
|
case 'SUP':
|
|
|
|
if ($strings[$i+1] != '(') {
|
|
|
|
$this->sup_classes->push(preg_replace("/'/",'',$strings[++$i]));
|
|
|
|
|
|
|
|
} else {
|
|
|
|
$i++;
|
|
|
|
|
|
|
|
do {
|
|
|
|
$i++;
|
|
|
|
|
|
|
|
if ($strings[$i] != '$')
|
|
|
|
$this->sup_classes->push(preg_replace("/'/",'',$strings[$i]));
|
|
|
|
|
|
|
|
} while (! preg_match('/\)+\)?/',$strings[$i+1]));
|
|
|
|
}
|
|
|
|
|
2024-12-30 22:41:28 +11:00
|
|
|
if (static::DEBUG_VERBOSE)
|
|
|
|
Log::debug(sprintf('- Case SUP returned (%s)',$this->sup_classes->join(',')));
|
2023-02-14 21:38:42 +11:00
|
|
|
break;
|
|
|
|
|
|
|
|
case 'ABSTRACT':
|
|
|
|
$this->type = self::OC_ABSTRACT;
|
|
|
|
|
2024-12-30 22:41:28 +11:00
|
|
|
if (static::DEBUG_VERBOSE)
|
|
|
|
Log::debug(sprintf('- Case ABSTRACT returned (%s)',$this->type));
|
2023-02-14 21:38:42 +11:00
|
|
|
break;
|
|
|
|
|
|
|
|
case 'STRUCTURAL':
|
|
|
|
$this->type = self::OC_STRUCTURAL;
|
|
|
|
|
2024-12-30 22:41:28 +11:00
|
|
|
if (static::DEBUG_VERBOSE)
|
|
|
|
Log::debug(sprintf('- Case STRUCTURAL returned (%s)',$this->type));
|
2023-02-14 21:38:42 +11:00
|
|
|
break;
|
|
|
|
|
|
|
|
case 'AUXILIARY':
|
|
|
|
$this->type = self::OC_AUXILIARY;
|
|
|
|
|
2024-12-30 22:41:28 +11:00
|
|
|
if (static::DEBUG_VERBOSE)
|
|
|
|
Log::debug(sprintf('- Case AUXILIARY returned (%s)',$this->type));
|
2023-02-14 21:38:42 +11:00
|
|
|
break;
|
|
|
|
|
|
|
|
case 'MUST':
|
|
|
|
$attrs = collect();
|
|
|
|
|
|
|
|
$i = $this->parseList(++$i,$strings,$attrs);
|
|
|
|
|
2024-12-30 22:41:28 +11:00
|
|
|
if (static::DEBUG_VERBOSE)
|
|
|
|
Log::debug(sprintf('= parseList returned %d (%s)',$i,$attrs->join(',')));
|
2023-02-14 21:38:42 +11:00
|
|
|
|
|
|
|
foreach ($attrs as $string) {
|
|
|
|
$attr = new ObjectClassAttribute($string,$this->name);
|
|
|
|
|
|
|
|
if ($server->isForceMay($attr->getName())) {
|
|
|
|
$this->may_force->push($attr);
|
|
|
|
$this->may_attrs->push($attr);
|
|
|
|
|
|
|
|
} else
|
|
|
|
$this->must_attrs->push($attr);
|
|
|
|
}
|
|
|
|
|
2024-12-30 22:41:28 +11:00
|
|
|
if (static::DEBUG_VERBOSE)
|
|
|
|
Log::debug(sprintf('- Case MUST returned (%s) (%s)',$this->must_attrs->join(','),$this->may_force->join(',')));
|
2023-02-14 21:38:42 +11:00
|
|
|
break;
|
|
|
|
|
|
|
|
case 'MAY':
|
|
|
|
$attrs = collect();
|
|
|
|
|
|
|
|
$i = $this->parseList(++$i,$strings,$attrs);
|
|
|
|
|
2024-12-30 22:41:28 +11:00
|
|
|
if (static::DEBUG_VERBOSE)
|
|
|
|
Log::debug(sprintf('parseList returned %d (%s)',$i,$attrs->join(',')));
|
2023-02-14 21:38:42 +11:00
|
|
|
|
|
|
|
foreach ($attrs as $string) {
|
|
|
|
$attr = new ObjectClassAttribute($string,$this->name);
|
|
|
|
$this->may_attrs->push($attr);
|
|
|
|
}
|
|
|
|
|
2024-12-30 22:41:28 +11:00
|
|
|
if (static::DEBUG_VERBOSE)
|
|
|
|
Log::debug(sprintf('- Case MAY returned (%s)',$this->may_attrs->join(',')));
|
2023-02-14 21:38:42 +11:00
|
|
|
break;
|
|
|
|
|
|
|
|
default:
|
|
|
|
if (preg_match('/[\d\.]+/i',$strings[$i]) && ($i === 1)) {
|
|
|
|
$this->oid = $strings[$i];
|
2024-12-30 22:41:28 +11:00
|
|
|
if (static::DEBUG_VERBOSE)
|
|
|
|
Log::debug(sprintf('- Case default returned (%s)',$this->oid));
|
2023-02-14 21:38:42 +11:00
|
|
|
|
|
|
|
} elseif ($strings[$i])
|
|
|
|
Log::alert(sprintf('! Case default discovered a value NOT parsed (%s)',$strings[$i]),['line'=>$line]);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
public function __get(string $key): mixed
|
|
|
|
{
|
|
|
|
switch ($key) {
|
2023-09-02 20:28:04 +10:00
|
|
|
case 'attributes':
|
|
|
|
return $this->getAllAttrs();
|
|
|
|
|
2023-02-14 21:38:42 +11:00
|
|
|
case 'sup':
|
|
|
|
return $this->sup_classes;
|
|
|
|
|
|
|
|
case 'type_name':
|
|
|
|
switch ($this->type) {
|
|
|
|
case self::OC_STRUCTURAL: return 'Structural';
|
|
|
|
case self::OC_ABSTRACT: return 'Abstract';
|
|
|
|
case self::OC_AUXILIARY: return 'Auxiliary';
|
|
|
|
default:
|
|
|
|
throw new InvalidUsage('Unknown ObjectClass Type: '.$this->type);
|
|
|
|
}
|
|
|
|
|
|
|
|
default: return parent::__get($key);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2023-09-02 20:28:04 +10:00
|
|
|
/**
|
|
|
|
* Return a list of attributes that this objectClass provides
|
|
|
|
*
|
|
|
|
* @return Collection
|
|
|
|
*/
|
|
|
|
public function getAllAttrs(): Collection
|
|
|
|
{
|
|
|
|
return $this->getMustAttrs()->merge($this->getMayAttrs());
|
|
|
|
}
|
|
|
|
|
2023-02-14 21:38:42 +11:00
|
|
|
/**
|
|
|
|
* Adds an objectClass to the list of objectClasses that inherit
|
|
|
|
* from this objectClass.
|
|
|
|
*
|
|
|
|
* @param String $name The name of the objectClass to add
|
|
|
|
*/
|
|
|
|
public function addChildObjectClass(string $name): void
|
|
|
|
{
|
|
|
|
if ($this->child_objectclasses->search($name) === FALSE) {
|
|
|
|
$this->child_objectclasses->push($name);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2023-09-02 20:28:04 +10:00
|
|
|
/**
|
|
|
|
* Returns the array of objectClass names which inherit from this objectClass.
|
|
|
|
*
|
|
|
|
* @return Collection Names of objectClasses which inherit from this objectClass.
|
|
|
|
* @deprecated use $this->child_objectclasses
|
|
|
|
*/
|
|
|
|
public function getChildObjectClasses(): Collection
|
|
|
|
{
|
|
|
|
return $this->child_objectclasses;
|
|
|
|
}
|
|
|
|
|
2023-02-14 21:38:42 +11:00
|
|
|
/**
|
|
|
|
* Behaves identically to addMustAttrs, but it operates on the MAY
|
|
|
|
* attributes of this objectClass.
|
|
|
|
*
|
|
|
|
* @param array $attr An array of attribute names (strings) to add.
|
|
|
|
*/
|
|
|
|
private function addMayAttrs(array $attr): void
|
|
|
|
{
|
|
|
|
if (! is_array($attr) || ! count($attr))
|
|
|
|
return;
|
|
|
|
|
|
|
|
$this->may_attrs = $this->may_attrs->merge($attr)->unique();
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Adds the specified array of attributes to this objectClass' list of
|
|
|
|
* MUST attributes. The resulting array of must attributes will contain
|
|
|
|
* unique members.
|
|
|
|
*
|
|
|
|
* @param array $attr An array of attribute names (strings) to add.
|
|
|
|
*/
|
|
|
|
private function addMustAttrs(array $attr): void
|
|
|
|
{
|
|
|
|
if (! is_array($attr) || ! count($attr))
|
|
|
|
return;
|
|
|
|
|
|
|
|
$this->must_attrs = $this->must_attrs->merge($attr)->unique();
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* @return Collection
|
|
|
|
* @deprecated use $this->may_force
|
|
|
|
*/
|
|
|
|
public function getForceMayAttrs(): Collection
|
|
|
|
{
|
|
|
|
return $this->may_force;
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Gets an array of AttributeType objects that entries of this ObjectClass may define.
|
|
|
|
* This differs from getMayAttrNames in that it returns an array of AttributeType objects
|
|
|
|
*
|
|
|
|
* @param bool $parents Also get the may attrs of our parents.
|
|
|
|
* @return Collection The array of allowed AttributeType objects.
|
|
|
|
*
|
|
|
|
* @throws InvalidUsage
|
|
|
|
* @see getMustAttrNames
|
|
|
|
* @see getMustAttrs
|
|
|
|
* @see getMayAttrNames
|
|
|
|
* @see AttributeType
|
|
|
|
*/
|
|
|
|
public function getMayAttrs(bool $parents=FALSE): Collection
|
|
|
|
{
|
|
|
|
// If we dont need our parents, then we'll just return ours.
|
|
|
|
if (! $parents)
|
|
|
|
return $this->may_attrs->sortBy(function($item) { return strtolower($item->name.$item->source); });
|
|
|
|
|
|
|
|
$attrs = $this->may_attrs;
|
|
|
|
|
|
|
|
foreach ($this->getParents() as $object_class) {
|
|
|
|
$sc = $this->server->schema('objectclasses',$object_class);
|
|
|
|
$attrs = $attrs->merge($sc->getMayAttrs($parents));
|
|
|
|
}
|
|
|
|
|
|
|
|
// Remove any duplicates
|
|
|
|
$attrs = $attrs->unique(function($item) { return $item->name; });
|
|
|
|
|
|
|
|
// Return a sorted list
|
|
|
|
return $attrs->sortBy(function($item) { return strtolower($item->name.$item->source); });
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Gets an array of attribute names (strings) that entries of this ObjectClass must define.
|
|
|
|
* This differs from getMayAttrs in that it returns an array of strings rather than
|
|
|
|
* array of AttributeType objects
|
|
|
|
*
|
|
|
|
* @param bool $parents An array of ObjectClass objects to use when traversing
|
|
|
|
* the inheritance tree. This presents some what of a bootstrapping problem
|
|
|
|
* as we must fetch all objectClasses to determine through inheritance which
|
|
|
|
* attributes this objectClass provides.
|
|
|
|
* @return Collection The array of allowed attribute names (strings).
|
|
|
|
*
|
|
|
|
* @throws InvalidUsage
|
|
|
|
* @see getMustAttrs
|
|
|
|
* @see getMayAttrs
|
|
|
|
* @see getMustAttrNames
|
|
|
|
*/
|
|
|
|
public function getMayAttrNames(bool $parents=FALSE): Collection
|
|
|
|
{
|
|
|
|
return $this->getMayAttrs($parents)->ppluck('name');
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Gets an array of AttributeType objects that entries of this ObjectClass must define.
|
|
|
|
* This differs from getMustAttrNames in that it returns an array of AttributeType objects
|
|
|
|
*
|
|
|
|
* @param bool $parents Also get the must attrs of our parents.
|
|
|
|
* @return Collection The array of required AttributeType objects.
|
|
|
|
*
|
|
|
|
* @throws InvalidUsage
|
|
|
|
* @see getMustAttrNames
|
|
|
|
* @see getMayAttrs
|
|
|
|
* @see getMayAttrNames
|
|
|
|
*/
|
|
|
|
public function getMustAttrs(bool $parents=FALSE): Collection
|
|
|
|
{
|
|
|
|
// If we dont need our parents, then we'll just return ours.
|
|
|
|
if (! $parents)
|
|
|
|
return $this->must_attrs->sortBy(function($item) { return strtolower($item->name.$item->source); });
|
|
|
|
|
|
|
|
$attrs = $this->must_attrs;
|
|
|
|
|
|
|
|
foreach ($this->getParents() as $object_class) {
|
|
|
|
$sc = $this->server->schema('objectclasses',$object_class);
|
|
|
|
$attrs = $attrs->merge($sc->getMustAttrs($parents));
|
|
|
|
}
|
|
|
|
|
|
|
|
// Remove any duplicates
|
|
|
|
$attrs = $attrs->unique(function($item) { return $item->name; });
|
|
|
|
|
|
|
|
// Return a sorted list
|
|
|
|
return $attrs->sortBy(function($item) { return strtolower($item->name.$item->source); });
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Gets an array of attribute names (strings) that entries of this ObjectClass must define.
|
|
|
|
* This differs from getMustAttrs in that it returns an array of strings rather than
|
|
|
|
* array of AttributeType objects
|
|
|
|
*
|
|
|
|
* @param bool $parents An array of ObjectClass objects to use when traversing
|
|
|
|
* the inheritance tree. This presents some what of a bootstrapping problem
|
|
|
|
* as we must fetch all objectClasses to determine through inheritance which
|
|
|
|
* attributes this objectClass provides.
|
|
|
|
* @return Collection The array of allowed attribute names (strings).
|
|
|
|
*
|
|
|
|
* @throws InvalidUsage
|
|
|
|
* @see getMustAttrs
|
|
|
|
* @see getMayAttrs
|
|
|
|
* @see getMayAttrNames
|
|
|
|
*/
|
|
|
|
public function getMustAttrNames(bool $parents=FALSE): Collection
|
|
|
|
{
|
|
|
|
return $this->getMustAttrs($parents)->ppluck('name');
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* This will return all our parent ObjectClass Objects
|
|
|
|
*/
|
|
|
|
public function getParents(): Collection
|
|
|
|
{
|
|
|
|
// If the only class is 'top', then we have no more parents
|
|
|
|
if (($this->sup_classes->count() === 1) && (strtolower($this->sup_classes->first()) === 'top'))
|
|
|
|
return collect();
|
|
|
|
|
|
|
|
$result = collect();
|
|
|
|
|
|
|
|
foreach ($this->sup_classes as $object_class) {
|
|
|
|
$result->push($object_class);
|
|
|
|
|
|
|
|
$oc = $this->server->schema('objectclasses',$object_class);
|
|
|
|
|
|
|
|
if ($oc)
|
|
|
|
$result = $result->merge($oc->getParents());
|
|
|
|
}
|
|
|
|
|
|
|
|
return $result;
|
|
|
|
}
|
|
|
|
|
2023-09-02 20:28:04 +10:00
|
|
|
/**
|
|
|
|
* Gets the objectClass names from which this objectClass inherits.
|
|
|
|
*
|
|
|
|
* @return Collection An array of objectClass names (strings)
|
|
|
|
* @deprecated use $this->sup_classes;
|
|
|
|
*/
|
|
|
|
public function getSupClasses(): Collection
|
|
|
|
{
|
|
|
|
return $this->sup_classes;
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Gets the type of this objectClass: STRUCTURAL, ABSTRACT, or AUXILIARY.
|
|
|
|
*
|
|
|
|
* @deprecated use $this->type_name
|
|
|
|
*/
|
|
|
|
public function getType()
|
|
|
|
{
|
|
|
|
return $this->type;
|
|
|
|
}
|
|
|
|
|
2023-02-14 21:38:42 +11:00
|
|
|
/**
|
|
|
|
* Determine if an array is listed in the may_force attrs
|
|
|
|
*/
|
|
|
|
public function isForceMay(string $attr): bool
|
|
|
|
{
|
|
|
|
return $this->may_force->ppluck('name')->contains($attr);
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Return if this objectClass is related to $oclass
|
|
|
|
*
|
|
|
|
* @param array $oclass ObjectClasses that this attribute may be related to
|
|
|
|
* @return bool
|
|
|
|
* @throws InvalidUsage
|
|
|
|
*/
|
|
|
|
public function isRelated(array $oclass): bool
|
|
|
|
{
|
|
|
|
// If I am in the array, we'll just return false
|
|
|
|
if (in_array_ignore_case($this->name,$oclass))
|
|
|
|
return FALSE;
|
|
|
|
|
|
|
|
foreach ($oclass as $object_class) {
|
|
|
|
$oc = $this->server->schema('objectclasses',$object_class);
|
|
|
|
|
|
|
|
if ($oc->isStructural() && in_array_ignore_case($this->name,$oc->getParents()))
|
|
|
|
return TRUE;
|
|
|
|
}
|
|
|
|
|
|
|
|
return FALSE;
|
|
|
|
}
|
|
|
|
|
|
|
|
public function isStructural(): bool
|
|
|
|
{
|
|
|
|
return $this->type === self::OC_STRUCTURAL;
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Parse an LDAP schema list
|
|
|
|
*
|
|
|
|
* A list starts with a ( followed by a list of attributes separated by $ terminated by )
|
|
|
|
* The first token can therefore be a ( or a (NAME or a (NAME)
|
|
|
|
* The last token can therefore be a ) or NAME)
|
|
|
|
* The last token may be terminated by more than one bracket
|
|
|
|
*/
|
|
|
|
private function parseList(int $i,array $strings,Collection &$attrs): int
|
|
|
|
{
|
|
|
|
$string = $strings[$i];
|
|
|
|
|
|
|
|
if (! preg_match('/^\(/',$string)) {
|
|
|
|
// A bareword only - can be terminated by a ) if the last item
|
|
|
|
if (preg_match('/\)+$/',$string))
|
|
|
|
$string = preg_replace('/\)+$/','',$string);
|
|
|
|
|
|
|
|
$attrs->push($string);
|
|
|
|
|
|
|
|
} elseif (preg_match('/^\(.*\)$/',$string)) {
|
|
|
|
$string = preg_replace('/^\(/','',$string);
|
|
|
|
$string = preg_replace('/\)+$/','',$string);
|
|
|
|
|
|
|
|
$attrs->push($string);
|
|
|
|
|
|
|
|
} else {
|
|
|
|
// Handle the opening cases first
|
|
|
|
if ($string === '(') {
|
|
|
|
$i++;
|
|
|
|
|
|
|
|
} elseif (preg_match('/^\(./',$string)) {
|
|
|
|
$string = preg_replace('/^\(/','',$string);
|
|
|
|
$attrs->push($string);
|
|
|
|
$i++;
|
|
|
|
}
|
|
|
|
|
|
|
|
// Token is either a name, a $ or a ')'
|
|
|
|
// NAME can be terminated by one or more ')'
|
|
|
|
while (! preg_match('/\)+$/',$strings[$i])) {
|
|
|
|
$string = $strings[$i];
|
|
|
|
|
|
|
|
if ($string === '$') {
|
|
|
|
$i++;
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (preg_match('/\)$/',$string))
|
|
|
|
$string = preg_replace('/\)+$/','',$string);
|
|
|
|
else
|
|
|
|
$i++;
|
|
|
|
|
|
|
|
$attrs->push($string);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
$attrs = $attrs->sort();
|
|
|
|
|
|
|
|
return $i;
|
|
|
|
}
|
|
|
|
}
|