Minor schema cosmetic code fixes, more Attribute implementation from old pla, start of LDAP DN view/edit

This commit is contained in:
Deon George 2023-03-02 09:54:30 +11:00
parent 491f04cd5d
commit 64d1a09db4
15 changed files with 234 additions and 99 deletions

View File

@ -20,6 +20,12 @@ class Attribute
// Current and Old Values // Current and Old Values
protected Collection $values; protected Collection $values;
// Can this attribute be deleted
protected bool $deletable = FALSE;
// Is this attribute an internal attribute
protected bool $internal;
/* /*
protected $oldvalues = array(); protected $oldvalues = array();
@ -27,8 +33,6 @@ class Attribute
protected $min_value_count = -1; protected $min_value_count = -1;
protected $max_value_count = -1; protected $max_value_count = -1;
# Is the attribute internal
protected $internal = false;
# Has the attribute been modified # Has the attribute been modified
protected $modified = false; protected $modified = false;
# Is the attribute being deleted because of an object class removal # Is the attribute being deleted because of an object class removal
@ -100,6 +104,20 @@ class Attribute
*/ */
} }
public function __get(string $key): mixed
{
switch ($key) {
case 'name':
return $this->{$key};
case 'internal': return isset($this->{$key}) && $this->{$key};
case 'name_lc': return strtolower($this->name);
default:
throw new \Exception('Unknown key:'.$key);
}
}
/** /**
* Determine how we render this attribute's value * Determine how we render this attribute's value
* *
@ -903,4 +921,18 @@ class Attribute
debug_dump_backtrace(sprintf('Unknown JS request %s',$type),1); debug_dump_backtrace(sprintf('Unknown JS request %s',$type),1);
} }
*/ */
/**
* Return an instance of this attribute that is deletable.
* This is primarily used for rendering to know if to render delete options.
*
* @return Attribute
*/
public function deletable(): self
{
$clone = clone $this;
$clone->deletable = TRUE;
return $clone;
}
} }

View File

@ -7,6 +7,6 @@ use App\Classes\LDAP\Attribute;
/** /**
* Represents an attribute whose values are binary * Represents an attribute whose values are binary
*/ */
class Binary extends Attribute abstract class Binary extends Attribute
{ {
} }

View File

@ -9,20 +9,33 @@ use App\Classes\LDAP\Attribute\Binary;
*/ */
final class JpegPhoto extends Binary final class JpegPhoto extends Binary
{ {
public function __construct(string $name,array $values)
{
parent::__construct($name,$values);
$this->internal = FALSE;
}
public function __toString(): string public function __toString(): string
{ {
$result = '';
// We'll use finfo to try and figure out what type of image is stored // We'll use finfo to try and figure out what type of image is stored
$f = new \finfo; $f = new \finfo;
$result = '<table class="table table-borderless p-0 m-0"><tr>';
foreach ($this->values as $value) { foreach ($this->values as $value) {
switch ($x=$f->buffer($value,FILEINFO_MIME_TYPE)) { switch ($x=$f->buffer($value,FILEINFO_MIME_TYPE)) {
case 'image/jpeg': case 'image/jpeg':
default: default:
$result .= sprintf('<img class="jpegphoto" src="data:%s;base64, %s" />',$x,base64_encode($value)); $result .= sprintf('<td><img class="jpegphoto" src="data:%s;base64, %s" />%s</td>',
$x,
base64_encode($value),
$this->deletable ? sprintf('<br><span class="btn btn-sm btn-danger"><i class="fas fa-trash-alt"></i> %s</span>',__('Delete')) : '');
} }
} }
$result .= '</tr></table>';
return $result; return $result;
} }
} }

View File

@ -5,7 +5,7 @@ namespace App\Classes\LDAP\Attribute;
use Illuminate\Support\Arr; use Illuminate\Support\Arr;
use Illuminate\Support\Facades\Log; use Illuminate\Support\Facades\Log;
use App\Classes\LDAP\{Attribute}; use App\Classes\LDAP\Attribute;
/** /**
* This factory is used to return LDAP attributes as an object * This factory is used to return LDAP attributes as an object
@ -20,11 +20,13 @@ class Factory
* Map of attributes to appropriate class * Map of attributes to appropriate class
*/ */
public const map = [ public const map = [
'entryuuid' => Internal\EntryUUID::class,
'jpegphoto' => Binary\JpegPhoto::class, 'jpegphoto' => Binary\JpegPhoto::class,
'supportedcontrol' => OID::class, 'objectclass' => ObjectClass::class,
'supportedextension' => OID::class, 'supportedcontrol' => Schema\OID::class,
'supportedfeatures' => OID::class, 'supportedextension' => Schema\OID::class,
'supportedsaslmechanisms' => Mechanisms::class, 'supportedfeatures' => Schema\OID::class,
'supportedsaslmechanisms' => Schema\Mechanisms::class,
]; ];
/** /**
@ -36,7 +38,7 @@ class Factory
*/ */
public static function create(string $attribute,array $values): Attribute public static function create(string $attribute,array $values): Attribute
{ {
$class = Arr::get(self::map,strtolower($attribute),Attribute::class); $class = Arr::get(self::map,$attribute,Attribute::class);
Log::debug(sprintf('%s:Creating LDAP Attribute [%s] as [%s]',static::LOGKEY,$attribute,$class)); Log::debug(sprintf('%s:Creating LDAP Attribute [%s] as [%s]',static::LOGKEY,$attribute,$class));
return new $class($attribute,$values); return new $class($attribute,$values);

View File

@ -0,0 +1,13 @@
<?php
namespace App\Classes\LDAP\Attribute;
use App\Classes\LDAP\Attribute;
/**
* Represents an attribute whose values are internal
*/
abstract class Internal extends Attribute
{
protected bool $internal = TRUE;
}

View File

@ -0,0 +1,12 @@
<?php
namespace App\Classes\LDAP\Attribute\Internal;
use App\Classes\LDAP\Attribute\Internal;
/**
* Represents an attribute whose values are binary
*/
final class EntryUUID extends Internal
{
}

View File

@ -0,0 +1,16 @@
<?php
namespace App\Classes\LDAP\Attribute;
use App\Classes\LDAP\Attribute;
/**
* Represents an attribute whose values are jpeg pictures
*/
final class ObjectClass extends Attribute
{
public function __toString(): string
{
return $this->values->sort()->join('<br>');
}
}

View File

@ -0,0 +1,50 @@
<?php
namespace App\Classes\LDAP\Attribute;
use Illuminate\Support\Arr;
use Illuminate\Support\Facades\Cache;
use App\Classes\LDAP\Attribute;
/**
* Represents an attribute whose values are schema related
*/
abstract class Schema extends Attribute
{
protected bool $internal = TRUE;
protected static function _get(string $filename,string $string,string $key): ?string
{
$array = Cache::remember($filename,86400,function() use ($filename) {
try {
$f = fopen($filename,'r');
} catch (\Exception $e) {
return NULL;
}
$result = collect();
while (! feof($f)) {
$line = trim(fgets($f));
if (! $line OR preg_match('/^#/',$line))
continue;
$fields = explode(':',$line);
$result->put($x=Arr::get($fields,0),[
'title'=>Arr::get($fields,1,$x),
'ref'=>Arr::get($fields,2),
'desc'=>Arr::get($fields,3,__('No description available, can you help with one?')),
]);
}
fclose($f);
return $result;
});
return Arr::get(($array ? $array->get($string) : []),$key);
}
}

View File

@ -1,16 +1,13 @@
<?php <?php
namespace App\Classes\LDAP\Attribute; namespace App\Classes\LDAP\Attribute\Schema;
use Illuminate\Support\Arr; use App\Classes\LDAP\Attribute\Schema;
use Illuminate\Support\Facades\Cache;
use App\Classes\LDAP\Attribute;
/** /**
* Represents an attribute whose values are binary * Represents an attribute whose values are binary
*/ */
class Mechanisms extends Attribute final class Mechanisms extends Schema
{ {
public function __toString(): string public function __toString(): string
{ {
@ -44,37 +41,8 @@ class Mechanisms extends Attribute
* @param string $key The title|ref|desc to return * @param string $key The title|ref|desc to return
* @return string|NULL * @return string|NULL
*/ */
private static function get(string $string,string $key): ?string protected static function get(string $string,string $key): ?string
{ {
$array = Cache::remember('saslmechanisms',86400,function() { return parent::_get(config_path('ldap_supported_saslmechanisms.txt'),$string,$key);
try {
$f = fopen(config_path('ldap_supported_saslmechanisms.txt'),'r');
} catch (\Exception $e) {
return NULL;
}
$result = collect();
while (! feof($f)) {
$line = trim(fgets($f));
if (! $line OR preg_match('/^#/',$line))
continue;
$fields = explode(':',$line);
$result->put($x=Arr::get($fields,0),[
'title'=>Arr::get($fields,1,$x),
'ref'=>Arr::get($fields,2),
'desc'=>Arr::get($fields,3,__('No description available, can you help with one?')),
]);
}
fclose($f);
return $result;
});
return Arr::get(($array ? $array->get($string) : []),$key);
} }
} }

View File

@ -1,16 +1,13 @@
<?php <?php
namespace App\Classes\LDAP\Attribute; namespace App\Classes\LDAP\Attribute\Schema;
use Illuminate\Support\Arr; use App\Classes\LDAP\Attribute\Schema;
use Illuminate\Support\Facades\Cache;
use App\Classes\LDAP\Attribute;
/** /**
* Represents an attribute whose values are binary * Represents an attribute whose values are binary
*/ */
class OID extends Attribute final class OID extends Schema
{ {
public function __toString(): string public function __toString(): string
{ {
@ -44,42 +41,13 @@ class OID extends Attribute
* ] * ]
* </code> * </code>
* *
* @param string $oid The OID number (ie, "1.3.6.1.4.1.4203.1.5.1") of the OID of interest. * @param string $string The OID number (ie, "1.3.6.1.4.1.4203.1.5.1") of the OID of interest.
* @param string $key The title|ref|desc to return * @param string $key The title|ref|desc to return
* @return string|null * @return string|null
* @testedby TranslateOidTest::testRootDSE() * @testedby TranslateOidTest::testRootDSE()
*/ */
private static function get(string $string,string $key): ?string protected static function get(string $string,string $key): ?string
{ {
$array = Cache::remember('oids',86400,function() { return parent::_get(config_path('ldap_supported_oids.txt'),$string,$key);
try {
$f = fopen(config_path('ldap_supported_oids.txt'),'r');
} catch (\Exception $e) {
return NULL;
}
$result = collect();
while (! feof($f)) {
$line = trim(fgets($f));
if (! $line OR preg_match('/^#/',$line))
continue;
$fields = explode(':',$line);
$result->put($x=Arr::get($fields,0),[
'title'=>Arr::get($fields,1,$x),
'ref'=>Arr::get($fields,2),
'desc'=>Arr::get($fields,3,__('No description available, can you help with one?')),
]);
}
fclose($f);
return $result;
});
return Arr::get(($array ? $array->get($string) : []),$key);
} }
} }

View File

@ -42,7 +42,7 @@ abstract class Base {
case 'oid': return $this->oid; case 'oid': return $this->oid;
default: default:
throw new InvalidUsage('Unknown key: '.$key); throw new InvalidUsage('Unknown key:'.$key);
} }
} }

View File

@ -3,6 +3,7 @@
namespace App\Ldap; namespace App\Ldap;
use Illuminate\Support\Arr; use Illuminate\Support\Arr;
use Illuminate\Support\Collection;
use LdapRecord\Models\Model; use LdapRecord\Models\Model;
use App\Classes\LDAP\Attribute\Factory; use App\Classes\LDAP\Attribute\Factory;
@ -14,6 +15,7 @@ class Entry extends Model
public function getAttributes(): array public function getAttributes(): array
{ {
$result = collect(); $result = collect();
foreach (parent::getAttributes() as $attribute => $value) { foreach (parent::getAttributes() as $attribute => $value) {
$result->put($attribute,Factory::create($attribute,$value)); $result->put($attribute,Factory::create($attribute,$value));
} }
@ -36,6 +38,13 @@ class Entry extends Model
/* METHODS */ /* METHODS */
public function getVisibleAttributes(): Collection
{
return collect($this->getAttributes())->filter(function($item) {
return ! $item->internal;
});
}
/** /**
* Return an icon for a DN based on objectClass * Return an icon for a DN based on objectClass
* *

View File

@ -10,14 +10,64 @@
@endsection @endsection
@section('main-content') @section('main-content')
<div class="bg-white p-3"> <div class="main-card mb-3 card">
<table class="table"> <div class="card-body">
@foreach ($o->getAttributes() as $attribute => $value) <div class="card-header-tabs">
<tr> <ul class="nav nav-tabs">
<th class="w-25">{{ $attribute }}</th> <li class="nav-item"><a data-toggle="tab" href="#attributes" class="nav-link active">{{ __('Attributes') }}</a></li>
<td>{!! $value !!}</td> <li class="nav-item"><a data-toggle="tab" href="#placeholder" class="nav-link">placeholder</a></li>
</tr> <li class="nav-item"><a data-toggle="tab" href="#internal" class="nav-link">{{ __('Internal') }}</a></li>
@endforeach <li class="nav-item"><a data-toggle="tab" href="#addtemplate" class="nav-link">{{ __('Add Template') }}</a></li>
</table> </ul>
<div class="tab-content">
<!-- All Attributes -->
<div class="tab-pane active" id="attributes" role="tabpanel">
<div class="row">
<div class="offset-2 col-8">
<table class="table">
@foreach ($o->getVisibleAttributes() as $ao)
<tr class="bg-light text-dark small">
<th class="w-25" colspan="2">
{{ $ao->name }}
<!-- Attribute Hints -->
<span class="float-right"></span>
</th>
</tr>
<tr>
<td class="pl-5">
{!! $ao->deletable() !!}<br>
<span class="p-0 m-0" id="add{{ $ao->name_lc }}"></span>
<span class="btn btn-sm btn-outline-primary mt-3 mb-3"><i class="fas fa-plus"></i> {{ __('Add Value') }}</span>
</td>
{{--
<td>@dump($ao)</td>
--}}
</tr>
@endforeach
</table>
</div>
</div>
</div>
<!-- Templates -->
<div class="tab-pane" id="placeholder" role="tabpanel">
<div><i class="fas fa-fw fa-spinner fa-pulse"></i></div>
</div>
<!-- Internal Attributes -->
<div class="tab-pane" id="internal" role="tabpanel">
<div><i class="fas fa-fw fa-spinner fa-pulse"></i></div>
</div>
<!-- Add Template -->
<div class="tab-pane" id="addtemplate" role="tabpanel">
<div><i class="fas fa-fw fa-spinner fa-pulse"></i></div>
</div>
</div>
</div>
</div>
</div> </div>
@endsection @endsection

View File

@ -12,14 +12,14 @@
@section('main-content') @section('main-content')
<div class="bg-white p-3"> <div class="bg-white p-3">
<table class="table"> <table class="table">
@foreach ($s->rootDSE()->getAttributes() as $attribute => $value) @foreach ($s->rootDSE()->getAttributes() as $attribute => $ao)
<tr> <tr>
<th class="w-25"> <th class="w-25">
{!! ($x=$s->schema('attributetypes',$attribute)) {!! ($x=$s->schema('attributetypes',$attribute))
? sprintf('<a class="attributetype" id="strtolower(%s)" href="%s">%s</a>',$x->name_lc,url('schema/attributetypes',$x->name_lc),$x->name) ? sprintf('<a class="attributetype" id="strtolower(%s)" href="%s">%s</a>',$x->name_lc,url('schema/attributetypes',$x->name_lc),$x->name)
: $attribute !!} : $attribute !!}
</th> </th>
<td>{!! $value !!}</td> <td>{!! $ao !!}</td>
</tr> </tr>
@endforeach @endforeach
</table> </table>

View File

@ -11,7 +11,9 @@
@section('main-content') @section('main-content')
<div class="main-card mb-3 card"> <div class="main-card mb-3 card">
<div class="card-body"><h5 class="card-title">{{ __('Schema Information') }}</h5> <div class="card-body">
<h5 class="card-title">{{ __('Schema Information') }}</h5>
<ul class="nav nav-tabs"> <ul class="nav nav-tabs">
<li class="nav-item"><a data-toggle="tab" href="#objectclasses" class="nav-link">{{ __('Object Classes') }}</a></li> <li class="nav-item"><a data-toggle="tab" href="#objectclasses" class="nav-link">{{ __('Object Classes') }}</a></li>
<li class="nav-item"><a data-toggle="tab" href="#attributetypes" class="nav-link">{{ __('Attribute Types') }}</a></li> <li class="nav-item"><a data-toggle="tab" href="#attributetypes" class="nav-link">{{ __('Attribute Types') }}</a></li>