# ----------------------------------------------------------------------------- =encoding utf8 =head1 NAME Quiq::ContentProcessor - Prozessor für Abschnitts-Dateien =head1 BASE CLASS L =head1 SYNOPSIS use Quiq::ContentProcessor; $cop = Quiq::ContentProcessor->new('.mytool'); $cop->registerType('MyTool::A','a','A','A'); $cop->registerType('MyTool::B','b','B','A'); ... $cop->load(@paths)->commit; for my $ent ($cop->entities) { $cop->msg($ent->name); } =head1 DESCRIPTION Ein Objekt der Klasse repräsentiert einen Prozessor für Entitäts-Dateien. Die Dateien bestehen aus einer Folge von Abschnitten, die von einem Abschnitts-Parser (s. Klasse Quiq::Section::Parser) geparsed und zu Abschnitts-Objekten instantiiert werden. Der Prozessor delegiert die Verarbeitung der Abschnitts-Objekte an die per registerType() registierten Entitäts-Klassen (Plugin-Schnittstelle). Diese bauen aus den Abschnitts-Objekten Entitäts-Strukturen auf, aus denen die Ausgabedateien generiert werden. Die Entitäts-Klassen sind nicht Teil des ContentProzessors. Bei Anwendung der Operation L() wird der Quelltext jeder Entität gegen den Stand im Storage verglichen. Im Falle einer Änderung wird die Entität als geändert gekennzeichnet, d.h. ihre Ausgabedateien müssen neu generiert werden. Das Resultat der Ausführung ist eine Menge von Entitäts-Objekten plus ihrem Änderungs-Status. Die Menge der Entitäts-Objekte kann mit der Methode L() abgefragt werden. =head2 Universelles Plugin Ein I kann definiert werden, indem bei L() nur $pluginClass und $extension als Argumente angegeben werden. An diese Plugin-Klasse werden alle (Haupt-)Abschnitts-Objekte delegiert, für die kein Plugin definiert ist. Logischerweise kann es höchstens ein Universelles Plugin geben. Für ein Universelles Plugin findet keine Attribut-Validierung in der Basisklassenmethode create() statt. Das Konzept ist in erster Linie für allgemeine Programme wie z.B. Testprogramme gedacht. =head2 Ausgaben Der Umfang an Ausgaben wird mit der Konstruktor-Option -verbosity=>$level eingestellt. Default ist 1. Die Methode msg() schreibt eine Ausgabe nach STDERR. Der erste Parameter gibt den Verbosity-Level an. Ist dieser größer als der eingestellte Verbosity-Level, unterbleibt die Ausgabe. $cop->msg(2,$text); Ist kein Level angegeben, erfolgt die Ausgabe I: $cop->msg($text); Der Meldungstext $text kann printf-Formatelemente enthalten, diese werden wie bei printf durch die zusätzlich angegebenen Argumente ersetzt: $cop->msg($text,@args); =head1 EXAMPLES Füge alle Entitäts-Definitionen im Storage zu einem einzigen Datenstrom zusammen und schreibe diesen nach STDOUT (z.B. für Refactoring): $cop->load->fetch('-'); Übertrage alle Entitäts-Definitionen im Storage in Verzeichnis $dir (das Verzeichnis hat die gleiche Struktur wie das Verzeichnis def im Storage): $cop->load->fetch->($dir); Liste alle Entitäten vom Typ $type auf: $cop->load; for my $ent ($cop->entities($type)) { $cop->msg($ent->name); } =cut # ----------------------------------------------------------------------------- package Quiq::ContentProcessor; use base qw/Quiq::Hash/; use v5.10; use strict; use warnings; our $VERSION = '1.197'; use Quiq::Path; use Quiq::Option; use Time::HiRes (); use Quiq::Perl; use Quiq::Hash; use Quiq::DestinationTree; use Quiq::Terminal; use Quiq::Section::Parser; use Quiq::Section::Object; use Quiq::Hash::Db; use Quiq::Formatter; # ----------------------------------------------------------------------------- =head1 METHODS =head2 Konstruktor =head3 new() - Instantiiere ContentProcessor-Objekt =head4 Synopsis $cop = $class->new($storage,@opt); =head4 Arguments =over 4 =item $storage Das Storage-Verzeichnis, z.B. '.storage'. =back =head4 Options =over 4 =item -verbosity => 0|1|2 (Default: 1) Umfang der Laufzeit-Meldungen. =back =head4 Returns ContentProcessor-Objekt =head4 Description Instantiiere ein ContentProcessor-Objekt und liefere eine Referenz auf dieses Objekt zurück. =cut # ----------------------------------------------------------------------------- sub new { my $class = shift; my $storage = Quiq::Path->expandTilde(shift); my $verbosity = 1; Quiq::Option->extract(\@_, -verbosity => \$verbosity, ); # Objekt instantiieren my $self = $class->SUPER::new( t0 => scalar Time::HiRes::gettimeofday, verbosity => $verbosity, storage => $storage, pluginA => [], fileA => [], parsedSections => 0, parsedLines => 0, parsedChars => 0, parsedBytes => 0, # memoize entityA => undef, entityTypeA => undef, extensionRegex => undef, needsUpdateDb => undef, needsTestDb => undef, typeH => undef, ); return $self; } # ----------------------------------------------------------------------------- =head2 Storage =head3 storage() - Pfad zum oder innerhalb des Storage =head4 Synopsis $path = $cop->storage; $path = $cop->storage($subPath); =head4 Arguments =over 4 =item $subPath Ein Sub-Pfad innerhalb des Storage. =back =head4 Returns Pfad =head4 Description Liefere den Pfad des Storage, ggf. ergänzt um den Sub-Pfad $subPath. =cut # ----------------------------------------------------------------------------- sub storage { my $self = shift; # @_: $subPath my $path = $self->{'storage'}; if (@_) { $path .= '/'.shift; } return $path; } # ----------------------------------------------------------------------------- =head2 Plugins =head3 registerType() - Registriere Entitäts-Typ =head4 Synopsis $cop->registerType($pluginClass,$extension,$entityType,$sectionType,@keyVal); $cop->registerType($pluginClass,$extension); # universelles Plugin =head4 Arguments =over 4 =item $pluginClass Name der Plugin-Klasse, z.B. 'Program::Shell'. =item $extension Datei-Erweiterung für Dateien dieses Typs, ohne Punkt, z.B. 'prg'. =item $entityType Entitätstyp-Bezeichner, z.B. 'Program' oder 'Class/Perl'. =item $sectionType Abschnitts-Bezeichner ohne Klammerung, z.B. 'Program'. =item @keyVal Abschnitts-Attribute, die über den Abschnitts-Bezeichner hinaus den Dateityp kennzeichnen, z.B. Language=>'Perl'. =back =head4 Returns nichts =head4 Description Registriere Plugin-Klasse $pluginClass für Abschnitts-Objekte des Typs $sectionType und den Eigenschaften @keyVal. Entsprechende Dateien werden an der Extension $extension erkannt. Als Typ-Bezeichner für Entitäten dieses Typs vereinbaren wir $entityType. Die Plugin-Klasse wird automatisch geladen, falls sie noch nicht vorhanden ist (sie kann für Tests also auch "inline", d.h. im Quelltext des rufenden Code, definiert werden). Die Plugin-Definition wird intern auf einem Hash-Objekt gespeichert, das vom ContentProcessor mit den instantiierten Entitäten verbunden wird. Es kann auch ein I definiert werden (siehe Abschnitt L). =cut # ----------------------------------------------------------------------------- sub registerType { my ($self,$pluginClass,$extension,$entityType,$sectionType,@keyVal) = @_; Quiq::Perl->loadClass($pluginClass); push @{$self->{'pluginA'}},Quiq::Hash->new( class => $pluginClass, extension => $extension, entityType => $entityType, sectionType => $sectionType, keyValA => \@keyVal, ); return; } # ----------------------------------------------------------------------------- =head3 entityTypes() - Liste der Abschnitts-Typen =head4 Synopsis @types | $typeA = $cop->entityTypes; =head4 Returns Liste von Abschnitts-Typen. Im Skalarkontext eine Referenz auf die Liste. =head4 Description Liefere die Liste der Abschnitts-Typen (Bezeichner), die per registerType() registriert wurden. =cut # ----------------------------------------------------------------------------- sub entityTypes { my $self = shift; my $a = $self->memoize('entityTypeA',sub { my ($self,$key) = @_; my %h; for my $plg (@{$self->{'pluginA'}}) { $h{$plg->entityType}++; } return [sort keys %h]; }); return wantarray? @$a: $a; } # ----------------------------------------------------------------------------- =head3 plugin() - Ermittele Plugin zu Abschnitts-Objekt =head4 Synopsis $plg = $cop->plugin($sec); =head4 Arguments =over 4 =item $sec Abschnitts-Objekt =back =head4 Returns Plugin-Objekt =head4 Description Ermittele das Plugin zu Abschnitts-Objekt $sec. Existiert kein Plugin zu dem Abschnitts-Objekt, liefere C. =cut # ----------------------------------------------------------------------------- my %Plugins; # Section-Type => Liste der Plugins my $Plugin; # Universelles Plugin (es sollte höchstens eins definiert sein) sub plugin { my ($self,$sec) = @_; if (!%Plugins) { # Indiziere Plugins nach Section-Type for my $plg (@{$self->{'pluginA'}}) { if (my $type = $plg->sectionType) { # Normales Plugin push @{$Plugins{$type}},$plg; } else { # Universelles Plugin $Plugin = $plg; } } } # Prüfe Section gemäß Plugin-Kriterien if (my $pluginA = $Plugins{$sec->type}) { for my $plg (@$pluginA) { my $ok = 1; my $a = $plg->keyValA; for (my $i = 0; $i < @$a; $i += 2) { if ($sec->get($a->[$i]) ne $a->[$i+1]) { $ok = 0; last; } } if ($ok) { return $plg; } } } # Kein Plugin zum SectionType gefunden. Wir liefern das # universelle Plugin, sofern existent, oder undef return $Plugin? $Plugin: undef; } # ----------------------------------------------------------------------------- =head2 Operationen =head3 init() - Erzeuge Storage =head4 Synopsis $cop = $cop->init; =head4 Returns ContentProcessor-Objekt (für Method-Chaining) =head4 Description Erzeuge den Storage, falls er nicht existiert. Existiert er bereits, hat der Aufruf keinen Effekt. =cut # ----------------------------------------------------------------------------- sub init { my $self = shift; $self->msg(1,'%T ==init'); my $storage = $self->storage; if (!-e $storage) { for ('','/db','/def','/pure') { my $dir = "$storage$_"; Quiq::Path->mkdir($dir); $self->msg(1,"$dir -- directory created"); } } return $self; } # ----------------------------------------------------------------------------- =head3 commit() - Übertrage Änderungen in den Storage =head4 Synopsis $cop = $cop->commit(@opt); =head4 Options =over 4 =item -incomplete => $bool (Default: 0) Lade fehlende Entitäten aus dem Storage nach anstatt sie zum Löschen anzubieten. =back =head4 Returns ContentProcessor-Objekt (für Method-Chaining) =head4 Description Vergleiche den Quelltext jeder Entität gegen den Quelltext im Storage und übertrage etwaige Änderungen dorthin. Geänderte Entitäten werden in der Datenbank als geändert gekennzeichnet. =cut # ----------------------------------------------------------------------------- sub commit { my $self = shift; # @_: @opt my $incomplete = 0; Quiq::Option->extract(\@_, -incomplete => \$incomplete, ); $self->msg(1,'%T ==commit=='); # Gleiche Eingabedateien mit Storage-Dateien ab my $defDir = $self->storage('def'); my $dt = Quiq::DestinationTree->new($defDir, -quiet => 0, -language => 'en', -outHandle => \*STDERR, ); $dt->addDir($defDir); for my $ent (@{$self->entities}) { # Schreibe Entity-Quelltext-Datei ins def-Verzeichnis, # falls sie differiert oder nicht existiert $dt->addFile($ent->entityFile($defDir),$ent->fileSourceRef, -encoding => 'utf-8', -skipEmptyFiles => 1, # Sub-Entities übergehen -onUpdate => sub { # Entität als geändert markieren $ent->needsUpdate(1); }, ); } $self->needsUpdateDb->sync; if ($incomplete) { # Lade fehlende Storage-Dateien nach my @files = grep {-f} $dt->obsoletePaths; if (@files) { $self->push(entityA=>$self->parseFiles(@files)); } } else { # Lösche fehlende Storage-Dateien (nach Rückfrage) $dt->cleanup(1,$self->getRef('t0')); } $dt->close; return $self; } # ----------------------------------------------------------------------------- =head3 load() - Lade Entitäts-Definitionen =head4 Synopsis $cop = $cop->load; $cop = $cop->load(@paths); =head4 Arguments =over 4 =item @paths Liste der Verzeichnisse und Dateien. Pfad '-' bedeutet STDIN. =back =head4 Returns ContentProcessor-Objekt (für Method-Chaining) =head4 Description Lade die Entitäts-Dateien der Pfade @paths. Ist @path leer, also kein Path angegeben, werden die Entitäts-Dateien aus dem Storage geladen. Die Methode kann beliebig oft aufgerufen werden, aber nur der erste Aufruf lädt die Dateien. Alle weiteren Aufrufe sind Null-Operationen. =cut # ----------------------------------------------------------------------------- sub load { my ($self,@paths) = @_; $self->memoize('entityA',sub { # Prüfen, ob Storage-Verzeichnis existert my $storage = $self->storage; if (!-d $storage) { die "ERROR: Directory '$storage' does not exist\n"; } $self->msg(1,'%T ==find=='); my @files; if (@paths) { # Wir laden die Dateien aus den angegebenen Pfaden. (Sub)Pfade, # die mit einem Punkt beginnen (.PATH) ignorieren wir in # diesem Fall, damit .jaz, .cotedo nicht konsultiert werden. for my $path (@paths) { if (-d $path) { push @files,$self->findFiles($path,-noDotPaths=>1); } else { push @files,$path; } } } else { # Wir laden die Dateien aus dem Storage, wenn kein Pfad # angegeben ist push @files,$self->findFiles($self->storage('def')); } my $n = scalar @files; if ($n > 1) { $self->msg(2,'Files: %N',$n); } $self->msg(1,'%T ==load=='); return scalar $self->parseFiles(@files); }); return $self; } # ----------------------------------------------------------------------------- =head3 fetchToDir() - Hole Entitäts-Definitionen aus dem Storage =head4 Synopsis $cop = $cop->fetchToDir($dir,$layout,@opt); =head4 Arguments =over 4 =item $dir Verzeichnis, in welches die Entitäts-Definitionen geschrieben werden. =item $layout Bezeichnung für das Verzeichnis-Layout. Dieser Parameter wird von filesToFetch() der Basisklasse nicht genutzt und ist daher hier nicht dokumentiert. Siehe Dokumentation bei den Subklassen. =back =head4 Options =over 4 =item -overwrite => $bool (Default: 0) Stelle keine Rückfrage, wenn Verzeichnis $dir bereits existiert. =back =head4 Returns ContentProcessor-Objekt (für Method-Chaining) =head4 Description Übertrage alle Entitäts-Definitionen in Verzeichnis $dir (oder nach STDOUT, s.u.) gemäß Layout $layout. Per Differenzbildung wird dabei ein konsistenter Stand hergestellt. Existiert Verzeichnis $dir nicht, wird es angelegt. Andernfalls wird eine Rückfrage gestellt, ob das Verzeichnis überschrieben werden soll (siehe auch Option --overwrite). Wird als Verzeichnis ein Bindestrich (-) angegeben, werden die Entitäts-Definitionen nach STDOUT geschrieben. Die Methode bezieht die zu schreibenden Dateien von der Methode L(), an die der Parameter $layout weiter gereicht wird. Die Methode kann in abgeleiteten Klassen überschrieben werden, um andere Strukturen zu generieren. =cut # ----------------------------------------------------------------------------- sub fetchToDir { my ($self,$dir,$layout) = @_; $self->msg(1,'%T ==fetch=='); # Optionen my $overwrite = 0; Quiq::Option->extract(\@_, -overwrite => \$overwrite, ); if ($dir eq '-') { for my $e ($self->filesToFetch($layout)) { # eof-Marker ergänzen my $ref = ref $e->[1]? $e->[1]: \$e->[1]; $$ref =~ s/\n+$//; $$ref .= "\n\n# eof\n"; print $$ref; } return $self; } elsif (-d $dir && !$overwrite) { my $answ = Quiq::Terminal->askUser( "Overwrite directory '$dir'?", -values => 'y/n', -default => 'y', -outHandle => \*STDERR, -timer => $self->getRef('t0'), ); if ($answ ne 'y') { return $self; } } my $dt = Quiq::DestinationTree->new( # FIXME: Hartcodierte Pfade vermeiden "$dir/Export", "$dir/Language", "$dir/Library", "$dir/Package", -quiet => 0, -language => 'en', -outHandle => \*STDERR, ); # $dt->addDir($dir); for my $e ($self->filesToFetch($layout)) { # eof-Marker ergänzen my $ref = ref $e->[1]? $e->[1]: \$e->[1]; $$ref =~ s/\n+$//; $$ref .= "\n\n# eof\n"; $dt->addFile("$dir/$e->[0]",$ref, -encoding => 'utf-8', -skipEmptyFiles => 1, # Sub-Entities übergehen (z.B. ProgramClass) ); } # Lösche überzählige Storage-Dateien (nach Rückfrage) $dt->cleanup(1,$self->getRef('t0')); $dt->close; return $self; } # ----------------------------------------------------------------------------- =head2 Hilfsmethoden =head3 filesToFetch() - Liste der Dateien für fetch =head4 Synopsis @files = $cop->filesToFetch; @files = $cop->filesToFetch($layout); =head4 Arguments =over 4 =item $layout Bezeichner für eine bestimmte Abfolge und/oder Inhalt der Dateien. =back =head4 Returns Array mit zweielementigen Arrays =head4 Description Liefere die Liste der Dateien, die von den Methode L() und [ANCHOR NOT FOUND]() geschrieben werden. Jede Datei wird durch ein zweielementiges Array repräsentiert, bestehend aus einem Datei-Pfad sowie dem Datei-Inhalt. Der Datei-Inhalt kann als String oder String-Referenz angegeben sein. Diese (Basisklassen-)Methode liefert für jede Entität die Datei-Definiton [$ent->entityFile, $ent->sourceRef] Damit erzeugt die Methode fetch() die gleiche Struktur wie der ContentProcessor im Storage-Verzeichnis def. Die Methode kann in abgeleiteten Klassen überschrieben werden, um die Verzeichnisstruktur zu ändern und/oder den Inhalt der Dateien anders zusammenzustellen (z.B. mehrere Entity-Definitionen in einer Datei zusammenzufassen). In abgeleiteten Klassen können verschiedene Layouts durch das Argument $layout unterschieden werden. =cut # ----------------------------------------------------------------------------- sub filesToFetch { my ($self,$layout) = @_; my @files; for my $ent (@{$self->entities}) { push @files,[$ent->entityFile,$ent->fileSourceRef]; } return @files; } # ----------------------------------------------------------------------------- =head2 Eingabedateien =head3 extensionRegex() - Regex zum Auffinden von Eingabe-Dateien =head4 Synopsis $regex = $cop->extensionRegex; =head4 Returns Kompilierter Regex =head4 Description Liefere den regulären Ausdruck, der die Dateinamen matcht, die vom ContentProcessor verarbeitet werden. Der Regex wird genutzt, wenn ein I nach Eingabe-Dateien durchsucht wird. =cut # ----------------------------------------------------------------------------- sub extensionRegex { my $self = shift; return $self->memoize('extensionRegex',sub { my ($self,$key) = @_; # Dateinamen-Erweiterungen ermitteln. Verschiedene # Plugin-Klassen können identische Datei-Erweiterungen haben, # deswegen filtern wir über einen Hash. my %extension; for my $plg (@{$self->{'pluginA'}}) { # Plugin-Extensions $extension{$plg->extension}++; } # Regex erstellen my $regex; for (sort keys %extension) { my $ext = $_; if ($regex) { $regex .= '|'; } $regex .= $ext; } return qr/\.($regex)$/; }); } # ----------------------------------------------------------------------------- =head3 findFiles() - Finde Entitäts-Dateien in Verzeichnis =head4 Synopsis @files | $fileA = $cop->findFiles($dir,@opt); =head4 Arguments =over 4 =item $dir Verzeichnis, das nach Dateien durchsucht wird. =back =head4 Options =over 4 =item -noDotPaths => $bool (Default: 0) Lasse Pfade, deren letzte Komponente mit einem Punkt beginnt, aus. Auf diese Weise verhindern wir, dass Dot-Verzeichnissse durchsucht werden. =back =head4 Returns Liste der Datei-Pfade (Strings). Im Skalarkontext eine Referenz auf die Liste. =head4 Description Durchsuche Verzeichnis $dir nach Entitäts-Dateien unter Verwendung des Regex, der von L() geliefert wird. =cut # ----------------------------------------------------------------------------- sub findFiles { my ($self,$dir) = @_; # @_: @opt # Optionen my $noDotPaths = 0; Quiq::Option->extract(\@_, -noDotPaths => \$noDotPaths, ); # Operation ausführen my @files = Quiq::Path->find($dir, -type => 'f', -pattern => $self->extensionRegex, # FIMXE: ../PATH zulassen -exclude => $noDotPaths? qr{(^|/)\.[^/]+$}: undef, -sloppy => 1, ); return wantarray? @files: \@files; } # ----------------------------------------------------------------------------- =head3 parseFiles() - Parse Dateien zu Entitäten =head4 Synopsis @entities | $entityA = $cop->parseFiles(@files); =head4 Arguments =over 4 =item @files Liste der Dateien. '-' bedeutet STDIN. =back =head4 Returns Liste der Entitäten. Im Skalarkontext eine Referenz auf die Liste. =cut # ----------------------------------------------------------------------------- sub parseFiles { my ($self,@files) = @_; # Instantiiere Parser my $par = Quiq::Section::Parser->new( encoding => 'utf-8', ); # Parse Dateien zu Entitäten my (@entities,$fileEnt,$mainSec); for my $file (@files) { $par->parse($file,sub { my $sec = Quiq::Section::Object->new(@_); # $sec->removeEofMarker; my $brackets = $sec->brackets; if ($brackets eq '[]') { $fileEnt = $mainSec = $sec; } elsif ($brackets eq '()') { $mainSec = $sec; } while (1) { if ($sec->brackets eq '[]') { # Wandele Abschnitts-Objekt in Entität my $plg = $self->plugin($sec); if (!$plg) { # Fehler: Für jeden Haupt-Abschnitt muss ein # Plugin definiert worden sein $sec->error( 'COP-00001: Missing plugin for section', Section => $sec->fullType, ); } push @entities,$plg->class->create($sec,$self,$plg); } elsif (@entities) { # Verarbeite nächsten Sub-Abschnitt $self->processSubSection($fileEnt,$entities[-1], $mainSec,$sec); if ($sec->brackets eq '[]') { # Sub-Abschnitt als Entität verarbeiten # (Beispiel: ProgramClass => Class) redo; } } else { # Fehler: Erster Abschnitt ist kein []-Abschnitt $sec->error( 'COP-00002: First section must be a []-section', Section => $sec->fullType, ); } last; } # Datei-Entity um Abschnitts-Quelltext ergänzen $fileEnt->appendFileSource($sec); # Abschnittsobjekt gegen unabsichtliche Erweiterungen sperren $sec->lockKeys; return; }); # "# eof" vom Dateiende entfernen my $fileSourceR = $fileEnt->fileSourceRef; $$fileSourceR =~ s/\n+# eof\s*$/\n\n/; if (substr($$fileSourceR,-1,1) ne "\n") { $fileEnt->error( 'COP-00003: File entity must end with a newline', ); } } # Statistische Daten speichern $self->addNumber(parsedSections=>$par->get('parsedSections')); $self->addNumber(parsedLines=>$par->get('parsedLines')); $self->addNumber(parsedChars=>$par->get('parsedChars')); $self->addNumber(parsedBytes=>$par->get('parsedBytes')); return wantarray? @entities: \@entities; } # ----------------------------------------------------------------------------- =head3 processSubSection() - Verarbeite Sub-Abschnitt (abstrakt) =head4 Synopsis $cop->processSubSection($fileEnt,$ent,$mainSec,$sec); =head4 Arguments =over 4 =item $fileEnt Entität, die Träger des gesamten Quelltextes ist. Diese ist im Quelltext mit [] gekennzeichnet. =item $ent Die aktuelle Entität. Kann identisch zu $fileEnt oder eine Sub-Entität (z.B. (File) oder (ProgramClass)) sein. =item $mainSec Die Entität oder Sub-Entität, der die mit <> gekennzeichneten Abschnitte zugeordnet werden. =item $sec Der aktuelle Abschnitt. =back =head4 Returns nichts =cut # ----------------------------------------------------------------------------- sub processSubSection { my ($self,$fileEnt,$ent,$mainSec,$sec) = @_; return; } # ----------------------------------------------------------------------------- =head2 Entitäten =head3 entities() - Liste der geladenen Entities =head4 Synopsis @entities | $entityA = $cop->entities; @entities | $entityA = $cop->entities($type); =head4 Arguments =over 4 =item $type Abschnitts-Typ. =back =head4 Returns Liste von Entitäten. Im Skalarkontext eine Referenz auf die Liste. =head4 Description Liefere die Liste aller geladenen Entities oder aller geladenen Entities vom Typ $type. Bei der Abfrage der Entities eines Typs werden die Entities nach Name sortiert geliefert. =cut # ----------------------------------------------------------------------------- sub entities { my ($self,$type) = @_; if (!$type) { my $entityA = $self->{'entityA'}; return wantarray? @$entityA: $entityA; } my $h = $self->memoize('typeH',sub { my ($self,$key) = @_; my $typeA = $self->entityTypes; # Hash mit allen Abschnittstypen aufbauen my %h; for my $type (@$typeA) { $h{$type} ||= []; } # Entitäten zuordnen for my $ent (@{$self->{'entityA'}}) { push @{$h{$ent->entityType}},$ent; } # Entitäten nach Name sortieren for my $type (@$typeA) { @{$h{$type}} = sort {$a->name cmp $b->name} @{$h{$type}}; } return \%h; }); my $a = $h->{$type} || $self->throw( 'COP-00000: Unknown type', Type => $type, ); return wantarray? @$a: $a; } # ----------------------------------------------------------------------------- =head2 Persistente Information =head3 needsTestDb() - Persistenter Hash für Test-Status =head4 Synopsis $h = $cop->needsTestDb; =head4 Returns Hash-Referenz (persistenter Hash) =head4 Description Liefere eine Referenz auf den persistenten Hash, der den Status von Entitäten speichert. =cut # ----------------------------------------------------------------------------- sub needsTestDb { my $self = shift; return $self->memoize('needsTestDb',sub { my ($self,$key) = @_; my $file = $self->storage('db/entity-needsTest.db'); return Quiq::Hash::Db->new($file,'rw'); }); } # ----------------------------------------------------------------------------- =head3 needsUpdateDb() - Persistenter Hash für Entitäts-Status =head4 Synopsis $h = $cop->needsUpdateDb; =head4 Returns Hash-Referenz (persistenter Hash) =head4 Description Liefere eine Referenz auf den persistenten Hash, der den Status von Entitäten speichert. =cut # ----------------------------------------------------------------------------- sub needsUpdateDb { my $self = shift; return $self->memoize('needsUpdateDb',sub { my ($self,$key) = @_; my $file = $self->storage('db/entity-needsUpdate.db'); return Quiq::Hash::Db->new($file,'rw'); }); } # ----------------------------------------------------------------------------- =head2 Ausgabe von Information =head3 info() - Informationszeile =head4 Synopsis $str = $cop->info; =head4 Returns Zeichenkette =head4 Description Liefere eine Informationszeile mit statistischen Informationen, die am Ende der Verarbeitung ausgegeben werden kann. =cut # ----------------------------------------------------------------------------- sub info { my $self = shift; my $entityA = $self->get('entityA'); my $entityCount = $entityA? @$entityA: 0; return sprintf '%.3f sec; Entities: %s; Sections: %s; Lines: %s'. '; Bytes: %s; Chars: %s', Time::HiRes::gettimeofday-$self->get('t0'), Quiq::Formatter->readableNumber($entityCount,','), Quiq::Formatter->readableNumber($self->get('parsedSections'),','), Quiq::Formatter->readableNumber($self->get('parsedLines'),','), Quiq::Formatter->readableNumber($self->get('parsedBytes'),','), Quiq::Formatter->readableNumber($self->get('parsedChars'),','); } # ----------------------------------------------------------------------------- =head3 msg() - Gib Information aus =head4 Synopsis $cop->msg($text,@args); $cop->msg($level,$text,@args); =head4 Arguments =over 4 =item $level Der Verbosity-Level, ab dem die Information ausgegeben wird. =item $text Text, der ausgegeben werden soll. =item @args Argumente, die für printf-Platzhalter in $text eingesetzt werden. =back =head4 Returns nichts =head4 Description Gib Information $text auf STDERR aus, wenn $level kleinergleich dem beim Konstruktor vorgebenen Verbosity-Level ist. Der Text kann printf-Platzhalter enthalten, die durch die Argumente @args ersetzt werden. Zusätzlich wird der Platzhalter %T durch die aktuelle Ausführungsdauer in Sekunden ersetzt. =cut # ----------------------------------------------------------------------------- sub msg { my $self = shift; # @_: $level,$text,@args -or- $text,@args if ($_[0] =~ /^(\d+)$/ && shift > $self->{'verbosity'}) { # Keine Ausgabe, wenn der Meldungslevel höher ist # als der vorgegebene Verbosity-Level. return; } my $text = shift; $text =~ s/%T/sprintf '%.3f', Time::HiRes::gettimeofday-$self->get('t0')/e; $text =~ s/%N/Quiq::Formatter->readableNumber(shift,',')/eg; printf STDERR "$text\n",@_; return; } # ----------------------------------------------------------------------------- =head1 VERSION 1.197 =head1 AUTHOR Frank Seitz, L =head1 COPYRIGHT Copyright (C) 2021 Frank Seitz =head1 LICENSE This code is free software; you can redistribute it and/or modify it under the same terms as Perl itself. =cut # ----------------------------------------------------------------------------- 1; # eof