wiki-archive/twiki/lib/TWiki/Meta.pm

564 lines
14 KiB
Perl

# Module of TWiki Enterprise Collaboration Platform, http://TWiki.org/
#
# Copyright (C) 2001-2007 Peter Thoeny, peter@thoeny.org
# and TWiki Contributors. All Rights Reserved. TWiki Contributors
# are listed in the AUTHORS file in the root of this distribution.
# NOTE: Please extend that file, not this notice.
#
# This program is free software; you can redistribute it and/or
# modify it under the terms of the GNU General Public License
# as published by the Free Software Foundation; either version 2
# of the License, or (at your option) any later version. For
# more details read LICENSE in the root of this distribution.
#
# This program is distributed in the hope that it will be useful,
# but WITHOUT ANY WARRANTY; without even the implied warranty of
# MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.
#
# As per the GPL, removal of this notice is prohibited.
=pod
---+ package TWiki::Meta
Meta-data handling.
A meta-data object is a hash of different types of meta-data (keyed on
the type, such as 'FIELD' and 'TOPICINFO').
Each entry in the hash is an array, where each entry in the array
contains another hash of the key=value pairs, corresponding to a
single meta-datum.
If there may be multiple entries of the same top-level type (i.e. for FIELD
and FILEATTACHMENT) then the array hash multiple entries. These types
are referred to as "keyed" types. The array entries are keyed with the
attribute 'name' which must be in each entry in the array.
For unkeyed types, the array has only one entry.
The module knows nothing about how meta-data is stored. That is entirely the
responsibility of the Store module.
Meta-data objects are created by the Store engine when topics are read. They
are populated using the =put= method.
=cut
package TWiki::Meta;
use strict;
use Error qw(:try);
use Assert;
use TWiki::Merge;
=pod
---++ ClassMethod new($session, $web, $topic)
Construct a new, empty Meta collection.
=cut
sub new {
my ( $class, $session, $web, $topic ) = @_;
ASSERT($session->isa( 'TWiki')) if DEBUG;
my $this = bless( {}, $class );
# Note: internal fields must be prepended with _. All other
# fields will be assumed to be meta-data.
$this->{_session} = $session;
ASSERT($web) if DEBUG;
ASSERT($topic) if DEBUG;
$this->{_web} = $web;
$this->{_topic} = $topic;
$this->{FILEATTACHMENT} = [];
return $this;
}
=pod
---++ ClassMethod web()
Get the web name
=cut
sub web {
return $_[0]->{_web};
}
=pod
---++ ClassMethod topic()
Get the topic name
=cut
sub topic {
return $_[0]->{_topic};
}
=pod
---++ ObjectMethod put($type, \%args)
Put a hash of key=value pairs into the given type set in this meta.
See the main comment for this package to understand how meta-data is
represented.
=cut
sub put {
my( $this, $type, $args ) = @_;
ASSERT($this->isa( 'TWiki::Meta')) if DEBUG;
my $data = $this->{$type};
if( $data ) {
# overwrite old single value
$data->[0] = $args;
} else {
push( @{$this->{$type}}, $args );
}
}
=pod
---++ ObjectMethod putKeyed($type, \%args)
Put a hash of key=value pairs into the given type set in this meta. The
entries are keyed by 'name'.
See the main comment for this package to understand how meta-data is
represented.
=cut
# Note: Array is used instead of a hash to preserve sequence
sub putKeyed {
my( $this, $type, $args ) = @_;
ASSERT($this->isa( 'TWiki::Meta')) if DEBUG;
my $data = $this->{$type};
if( $data ) {
my $keyName = $args->{name};
ASSERT( $keyName ) if DEBUG;
my $i = scalar( @$data );
while( $keyName && $i-- ) {
if( $data->[$i]->{name} eq $keyName ) {
$data->[$i] = $args;
return;
}
}
push @$data, $args;
} else {
push( @{$this->{$type}}, $args );
}
}
=pod
---++ ObjectMethod putAll
Replaces all the items of a given key with a new array
This is the logical inverse of the find method
=cut
sub putAll {
my( $this, $type, @array ) = @_;
ASSERT($this->isa( 'TWiki::Meta')) if DEBUG;
$this->{$type} = \@array;
}
=pod
---++ ObjectMethod get( $type, $key ) -> \%hash
Find the value of a meta-datum in the map. If the type is
keyed, the $key parameter is required to say _which_
entry you want. Otherwise it can be undef.
WARNING SMELL If key is undef but the type is keyed you get the FIRST entry
If you want all the keys of a given type use the 'find' method.
The result is a reference to the hash for the item.
=cut
sub get {
my( $this, $type, $keyValue ) = @_;
ASSERT($this->isa( 'TWiki::Meta')) if DEBUG;
my $data = $this->{$type};
if( $data ) {
if( defined $keyValue ) {
foreach my $item ( @$data ) {
return $item if( $item->{name} eq $keyValue );
}
} else {
return $data->[0];
}
}
return undef;
}
=pod
---++ ObjectMethod find ( $type ) -> @values
Get all meta data for a specific type
Returns the array stored for the type. This will be zero length
if there are no entries.
=cut
sub find {
my( $this, $type ) = @_;
ASSERT($this->isa( 'TWiki::Meta')) if DEBUG;
my $itemsr = $this->{$type};
my @items = ();
if( $itemsr ) {
@items = @$itemsr;
}
return @items;
}
=pod
---++ StaticMethod indexByKey
See tests/unit/MetaTests.pm for an example
The result is a hash the same as the array provided by find but keyed by the keyName.
NB. results are indeterminate if the key you choose is not unique in the find.
=cut
sub indexByKey {
my( $keyName, @array) = @_;
my %findKeyed = ();
foreach my $result (@array) {
my $key = $result->{$keyName};
$findKeyed{$key} = $result;
}
return %findKeyed;
}
=pod
Flattens a keyed hash structure, taking only the values.
Returns a hash.
See tests/unit/MetaTests.pm for an example
=cut
sub deindexKeyed {
my (%hash) =@_;
my @array = ();
foreach my $key (keys %hash) {
my $value = $hash{$key};
push @array, $value;
}
return @array;
}
=pod
---++ ObjectMethod remove ( $type, $key )
With no type, will remove all the contents of the object.
With a $type but no $key, will remove _all_ items of that type (so for example if $type were FILEATTACHMENT it would remove all of them)
With a $type and a $key it will remove only the specific item.
=cut
sub remove {
my( $this, $type, $keyValue ) = @_;
ASSERT($this->isa( 'TWiki::Meta')) if DEBUG;
if( $keyValue ) {
my $data = $this->{$type};
my @newData = ();
foreach my $item ( @$data ) {
if( $item->{name} ne $keyValue ) {
push @newData, $item;
}
}
$this->{$type} = \@newData;
} elsif( $type ) {
delete $this->{$type};
} else {
foreach my $entry ( keys %$this ) {
unless( $entry =~ /^_/ ) {
$this->remove( $entry );
}
}
}
}
=pod
---++ ObjectMethod copyFrom( $otherMeta, $type, $nameFilter )
Copy all entries of a type from another meta data set. This
will destroy the old values for that type, unless the
copied object doesn't contain entries for that type, in which
case it will retain the old values.
If $type is undef, will copy ALL TYPES.
If $nameFilter is defined (an RE), it will copy only data where
{name} matches $nameFilter.
SMELL: This is a shallow copy
=cut
sub copyFrom {
my( $this, $otherMeta, $type, $filter ) = @_;
ASSERT($this->isa( 'TWiki::Meta')) if DEBUG;
ASSERT($otherMeta->isa( 'TWiki::Meta')) if DEBUG;
if( $type ) {
foreach my $item ( @{$otherMeta->{$type}} ) {
if( !$filter || ( $item->{name} && $item->{name} =~ /$filter/ )) {
my %data;
foreach my $k ( keys %$item ) {
$data{$k} = $item->{$k};
}
push( @{$this->{$type}}, \%data );
}
}
} else {
foreach my $k ( keys %$otherMeta ) {
unless( $k =~ /^_/ ) {
$this->copyFrom( $otherMeta, $k );
}
}
}
}
=pod
---++ ObjectMethod count ( $type ) -> $integer
Return the number of entries of the given type that are in this meta set
=cut
sub count {
my( $this, $type ) = @_;
ASSERT($this->isa( 'TWiki::Meta')) if DEBUG;
my $data = $this->{$type};
return scalar @$data if( defined( $data ));
return 0;
}
=pod
---++ ObjectMethod getRevisionInfo($fromrev) -> ( $date, $author, $rev, $comment )
Try and get revision info from the meta information, or, if it is not
present, kick down to the Store module for the same information.
Returns ( $revDate, $author, $rev, $comment )
$rev is an integer revision number.
=cut
sub getRevisionInfo {
my( $this, $fromrev ) = @_;
ASSERT($this->isa( 'TWiki::Meta')) if DEBUG;
my $store = $this->{_session}->{store};
my $topicinfo = $this->get( 'TOPICINFO' );
my( $date, $author, $rev, $comment );
if( $topicinfo ) {
$date = $topicinfo->{date} ;
$author = $this->{_session}->{users}->findUser($topicinfo->{author});
$rev = $topicinfo->{version};
$rev =~ s/^\$Rev(:\s*\d+)?\s*\$$/0/; # parse out SVN keywords in doc
$rev =~ s/^\d+\.//;
$comment = '';
if ( !$fromrev || $rev eq $fromrev ) {
return( $date, $author, $rev, $comment );
}
}
# Different rev, or no topic info, delegate to Store
( $date, $author, $rev, $comment ) =
$store->getRevisionInfo( $this->{_web}, $this->{_topic}, $fromrev );
return( $date, $author, $rev, $comment );
}
=pod
---++ ObjectMethod merge( $otherMeta, $formDef )
* =$otherMeta= - a block of meta-data to merge with $this
* =$formDef= reference to a TWiki::Form that gives the types of the fields in $this
Merge the data in the other meta block.
* File attachments that only appear in one set are preserved.
* Form fields that only appear in one set are preserved.
* Form field values that are different in each set are text-merged
* We don't merge for field attributes or title
* Topic info is not touched
* The =mergeable= method on the form def is used to determine if that fields is mergeable. if it isn't, the value currently in meta will _not_ be changed.
=cut
sub merge {
my ( $this, $other, $formDef ) = @_;
my $data = $other->{FIELD};
if( $data ) {
foreach my $otherD ( @$data ) {
my $thisD = $this->get( 'FIELD', $otherD->{name} );
if ( $thisD && $thisD->{value} ne $otherD->{value} ) {
if( $formDef->isTextMergeable( $thisD->{name} )) {
my $merged = TWiki::Merge::merge2(
'A', $otherD->{value}, 'B', $thisD->{value},
qr/(\s+)/,
$this->{_session},
$formDef->getField( $thisD->{name} ) );
# SMELL: we don't merge attributes or title
$thisD->{value} = $merged;
}
} elsif ( !$thisD ) {
$this->putKeyed('FIELD', $otherD );
}
}
}
$data = $other->{FILEATTACHMENT};
if( $data ) {
foreach my $otherD ( @$data ) {
my $thisD = $this->get( 'FILEATTACHMENT', $otherD->{name} );
if ( !$thisD ) {
$this->putKeyed('FILEATTACHMENT', $otherD );
}
}
}
}
=pod
---++ ObjectMethod stringify( $types ) -> $string
Return a string version of the meta object. Uses \n to separate lines.
If $types is specified, return only types specified by that RE.
=cut
sub stringify {
my( $this, $types ) = @_;
my $s = '';
$types ||= qr/^[A-Z]+$/;
foreach my $type ( grep { /$types/ } keys %$this ) {
foreach my $item ( @{$this->{$type}} ) {
$s .= "$type: " .
join(' ', map{ "$_='".($item->{$_}||'')."'" }
sort keys %$item ) .
"\n";
}
}
return $s;
}
=pod
---++ ObjectMethod forEachSelectedValue( $types, $keys, \&fn, \%options )
Iterate over the values selected by the regular expressions in $types and
$keys.
* =$types= - regular expression matching the names of fields to be processed. Will default to qr/^[A-Z]+$/ if undef.
* =$keys= - regular expression matching the names of keys to be processed. Will default to qr/^[a-z]+$/ if undef.
Iterates over each value, calling =\&fn= on each, and replacing the value
with the result of \&fn.
\%options will be passed on to $fn, with the following additions:
* =_type= => the type name (e.g. "FILEATTACHMENT")
* =_key= => the key name (e.g. "user")
=cut
sub forEachSelectedValue {
my( $this, $types, $keys, $fn, $options ) = @_;
$types ||= qr/^[A-Z]+$/;
$keys ||= qr/^[a-z]+$/;
foreach my $type ( grep { /$types/ } keys %$this ) {
$options->{_type} = $type;
my $data = $this->{$type};
next unless $data;
foreach my $datum ( @$data ) {
foreach my $key ( grep { /$keys/ } keys %$datum ) {
$options->{_key} = $key;
$datum->{$key} = &$fn( $datum->{$key}, $options );
}
}
}
}
=pod
---++ ObjectMethod getParent() -> $parent
Gets the TOPICPARENT name.
=cut
sub getParent {
my( $this ) = @_;
my $value = '';
my $parent = $this->get( 'TOPICPARENT' );
$value = $parent->{name} if( $parent );
# Return empty string (not undef), if TOPICPARENT meta is broken
$value = '' if (!defined $value);
return $value;
}
=pod
---++ ObjectMethod getFormName() -> $formname
Returns the name of the FORM, or '' if none.
=cut
sub getFormName {
my( $this ) = @_;
my $aForm = $this->get( 'FORM' );
if( $aForm ) {
return $aForm->{name};
}
return '';
}
1;