package VCP::Dest::revml ;
=head1 NAME
VCP::Dest::revml - Outputs versioned files to a revml file
=head1 SYNOPSIS
revml:[<output-file>]
revml:[<output-file>] --dtd <revml.dtd>
revml:[<output-file>] --version <version>
revml:[<output-file>] --compress
revml:[<output-file>] --no-indent
=head1 DESCRIPTION
=head1 OPTIONS
=over
=item --dtd
--dtd=revml.dtd
--version=0.33
The --dtd (or --version) option causes the output to be checked against
a particular version of revml. This does I<not> cause output to be in
that version, but makes sure that output is compliant with that version.
=item --compress
Generate a gzipped revml file as output. Requires gzip be installed
on the machine.
If the output filename ends in ".gz", the output will be compressed
even if the --compress flag isn't present.
=item --no-indent
Makes the revml output with all the start tags flush left rather than
indented to indicate the tree structure.
=back
=head1 EXTERNAL METHODS
=over
=cut
$VERSION = 0.1 ;
@ISA = qw( VCP::Dest VCP::Utils::revml );
use VCP::Logger qw( pr_doing BUG );
use VCP::Utils qw( start_dir_rel2abs );
use VCP::Utils::revml;
BEGIN {
## Beginning vcpers might try running a command like "vcp" just
## to see what happens. Since RevML is not required for most
## vcp uses and XML::Parser requires a C compiler, vcp is often
## distributed without XML::Parser, so this message is to help
## steer hapless new users to the help system if the XML output
## modules are also not found.
## The exit(1) is to avoid untidy and scary
## "compilation failed in BEGIN" messages
VCP::Logger::pr( <<TOHERE ), exit 1 unless eval "require RevML::Writer";
RevML::Writer is not installed or loading properly on this system and
it is required to write RevML files. If writing RevML files is not
what you want to do, try
vcp help
TOHERE
}
use strict ;
use Carp ;
use Digest::MD5 ;
use Fcntl ;
use File::Basename;
use MIME::Base64 ;
use RevML::Doctype ;
use RevML::Writer ;
use Symbol ;
use UNIVERSAL qw( isa ) ;
use VCP::Debug qw( :debug ) ;
use VCP::Dest;
use VCP::Logger qw( lg );
use VCP::Rev qw( iso8601format );
use VCP::Utils qw( shell_quote empty );
use Text::Diff ;
#use base qw( VCP::Dest VCP::Utils::revml );
#use fields (
# 'DTD_OPTION', ## The --dtd or --version flag from the command line
# 'COMPRESS', ## flag indicating gzip'd compressed output
# 'DOCTYPE', ## revml doc type
# 'OUT_FH', ## The handle of the output file
# 'WRITER', ## The XML::AutoWriter instance write with
# 'SEEN_REV', ## Whether we've seen our first revision or not.
# 'NO_INDENT', ## output all flush left, not indented to indicate tree structure
# 'FILES', ## The files we care about. When we check these out from
# ## the source, we need to hold on to them so that we
# ## can perform diffs. These are keyed on rev_id and
# ## contain VCP::RefCountedFile objects (assuming
# ## get_source_file() returns that kind of object).
#) ;
=item new
Creates a new instance. The only parameter is '-dtd', which overrides
the default DTD found by searching for modules matching RevML::DTD:v*.pm.
Attempts to create the output file if one is specified.
=cut
sub new {
my $self = shift->SUPER::new;
my ( $spec, $options ) = @_ ;
$self->parse_revml_repo_spec( $spec )
unless empty $spec;
$self->parse_options( $options );
return $self;
}
sub options_spec {
my $self = shift ;
return (
$self->SUPER::options_spec,
'dtd|version=s' => \$self->{DTD_OPTION},
"compress" => \$self->{COMPRESS},
"no-indent" => \$self->{NO_INDENT},
);
}
sub init {
my $self = shift ;
$self->head_revs->open_db;
$self->{DOCTYPE} = RevML::Doctype->new(
$self->{DTD_OPTION},
main::bundled_files()
);
my $file_name = $self->repo_filespec;
$file_name = "-"
if empty $file_name;
$file_name = start_dir_rel2abs $file_name
unless $file_name eq "-";
lg "revml file name is ", $file_name;
# always un-compress if filename ends in ".gz"
my $gzip;
if ( $^O =~ /Win32/ ) {
$self->{COMPRESS} = 1 if $file_name =~ /\.gz$/i ;
$gzip = "gzip.exe";
}
else {
$self->{COMPRESS} = 1 if $file_name =~ /\.gz$/ ;
$gzip = "gzip";
}
if ( $file_name eq '-' ) {
if( $self->{COMPRESS} ) {
require Symbol ;
$self->{OUT_FH} = Symbol::gensym ;
open( $self->{OUT_FH}, "| $gzip" )
or die "$!: | gzip" ;
}
else {
$self->{OUT_FH} = \*STDOUT ;
}
## TODO: Check OUT_FH for writability when it's set to STDOUT
}
else {
require Symbol ;
$self->{OUT_FH} = Symbol::gensym ;
## TODO: Provide a '-f' force option
if( $self->{COMPRESS} ) {
my $out_name = shell_quote $file_name;
open( $self->{OUT_FH}, "| $gzip > $out_name" )
or die "$!: | gzip > $file_name" ;
}
else {
open( $self->{OUT_FH}, ">$file_name" ) or die "$!: '$file_name'" ;
}
}
## BUG: Can't undo this AFAIK, so we're permanently altering STDOUT
## if $out_name eq '-'.
binmode $self->{OUT_FH};
$self->writer(
RevML::Writer->new(
DOCTYPE => $self->{DOCTYPE},
OUTPUT => $self->{OUT_FH},
)
);
return $self ;
}
=item ui_set_revml_repo_spec
set the repo_spec, but die if no good.
this should be called from a ui handler that
will handle exceptions.
=cut
sub ui_set_revml_repo_spec {
my $self = shift ;
$self->parse_revml_repo_spec( @_ );
my $file = $self->repo_filespec;
die "Error: '-' signifies standard output, not a file name.\n"
if $file eq '-';
$file = start_dir_rel2abs $file ;
die "Error: '$file' is a directory.\n"
if -d $file;
if( -e $file ) {
die "Warning: '$file' exists and is writable.\n" if -w $file;
die "Error: '$file' exists, but is not writable!\n";
}
my $dirname = dirname( $file );
die "Error: directory '$dirname' not found!\n"
unless -d $dirname;
}
sub _emit_characters {
my ( $w, $buf ) = @_ ;
$w->setDataMode( 0 ) ;
## Note that we don't let XML munge \r to be \n!!
while ( $$buf =~ m{\G(?:
( [\x00-\x08\x0b-\x1f\x7f-\xff])
| ([^\x00-\x08\x0b-\x1f\x7f-\xff]*)
)}gx
) {
if ( defined $1 ) {
$w->char( "", code => sprintf( "0x%02x", ord $1 ) ) ;
}
else {
$w->characters( $2 ) ;
}
}
}
sub handle_rev {
my $self = shift ;
my $r ;
( $r ) = @_ ;
debug "got ", $r->as_string if debugging;
my $w = $self->writer ;
if ( ! $self->{SEEN_REV} ) {
## We don't write any XML until the first rev arrives, so an empty
## RevML input doc (with root node but no revs) results in no output.
## This is more for non-RevML inputs, so that an export that selects
## no new revisions generates no output, which is easy to test for.
$w->setDataMode( 1 ) unless $self->{NO_INDENT};
$w->xmlDecl ;
my $h = $self->header ;
## VCP::Source::revml passes through the original date. Other sources
## don't.
$w->time(
defined $h->{time}
? iso8601format $h->{time}
: iso8601format gmtime
) ;
$w->rep_type( $h->{rep_type} ) ;
$w->rep_desc( $h->{rep_desc} ) ;
$w->comment( $h->{comment} ) if defined $h->{comment};
$w->rev_root( $h->{rev_root} ) ;
}
$self->{SEEN_REV} = 1;
## TODO: get rid of revs that aren't needed. We should only need
## the most recent rev with a work path on each branch.
$self->revs->add( $r );
my $fn = $r->name ;
my $is_base_rev = $r->is_base_rev ;
## type and rev_id are not provided for VSS deletes
debug "emitting revml for ", $r->as_string
if debugging;
eval {
$w->start_rev( id => $r->id );
$w->name( $fn );
$w->source_name( $r->source_name );
$w->source_filebranch_id( $r->source_filebranch_id );
$w->source_repo_id( $r->source_repo_id );
$w->action( $r->action );
$w->type( $r->type ) if defined $r->type ;
$w->p4_info( $r->p4_info ) if defined $r->p4_info ;
$w->cvs_info( $r->cvs_info ) if defined $r->cvs_info ;
if( defined $r->branch_id || defined $r->source_branch_id ) {
$w->branch_id(
defined $r->branch_id ? $r->branch_id : ()
);
$w->source_branch_id(
defined $r->source_branch_id ? $r->source_branch_id : ()
);
}
$w->rev_id( $r->rev_id ) if defined $r->rev_id ;
$w->source_rev_id( $r->source_rev_id ) if defined $r->source_rev_id ;
if( defined $r->change_id || defined $r->source_change_id ) {
$w->change_id(
defined $r->change_id ? $r->change_id : ()
);
$w->source_change_id(
defined $r->source_change_id ? $r->source_change_id : ()
);
}
$w->time( iso8601format $r->time ) if defined $r->time ;
$w->mod_time( iso8601format $r->mod_time ) if defined $r->mod_time ;
$w->user_id( $r->user_id ) if defined $r->user_id;
## Sorted for readability & testability
$w->label( $_ ) for sort $r->labels ;
unless ( empty $r->comment ) {
$w->start_comment ;
my $c = $r->comment ;
_emit_characters( $w, \$c ) ;
$w->end_comment ;
$w->setDataMode( 1 ) unless $self->{NO_INDENT};
}
$w->previous_id( $r->previous_id ) if defined $r->previous_id;
$w->from_id( $r->from_id ) if defined $r->from_id;
## Sorted for readability & testability
$w->earlier_id( $_ ) for sort $r->earlier_ids ;
my $convert_crs = $^O =~ /Win32/ && ( $r->type || "" ) eq "text" ;
my $digestion ;
my $close_it ;
my $cp;
$cp = $r->get_source_file
if $r->is_real_rev;
$self->{FILES}->{$r->id} = $cp unless empty $cp;
if ( $is_base_rev ) {
sysopen( F, $cp, O_RDONLY ) or die "$!: $cp\n" ;
binmode F ;
$digestion = 1 ;
$close_it = 1 ;
}
elsif ( $r->is_placeholder_rev ) {
$digestion = 0;
}
elsif ( $r->action eq 'delete' ) {
}
else {
BUG "\$r->get_source_file() returned undef" unless defined $cp;
BUG "\$r->get_source_file() returned ''" unless length $cp;
sysopen( F, $cp, O_RDONLY ) or die "$!: '$cp'\n" ;
## need to binmode it so ^Z can pass through, need to do \r and
## \r\n -> \n conversion ourselves.
binmode F ;
$close_it = 1 ;
my $buf ;
my $read ;
my $total_char_count = 0 ;
my $bin_char_count = 0 ;
while ( 1 ) {
$read = sysread( F, $buf, 100_000 ) ;
die "$! reading $cp\n" unless defined $read ;
last unless $read ;
$bin_char_count += $buf =~ tr/\x00-\x08\x0b-\x1f\x7f-\xff// ;
$total_char_count += length $buf ;
} ;
sysseek( F, 0, 0 ) or die "$! seeking on $cp\n" ;
$buf = '' unless $read ;
## base64 generate 77 chars (including the newline) for every 57 chars
## of input. A '<char code="0x01" />' element is 20 chars.
my $legal_xml_char_count = $total_char_count - $bin_char_count;
## TODO: calculate the diff no matter what in to a temp file,
## keeping stats on how to encode it (none vs. base64).
## If encoding it would be shorter than the encoded content of
## the new rev, use it. This should be safe now that we're using
## all-Perl diff & patch (which don't mind NUL, for instance).
## if both are eaual, use the more human readable format ("none")
my $encoding =
( $legal_xml_char_count + $bin_char_count * 20 )
<= $total_char_count * 77/57
? "none"
: "base64";
my $pr;
unless ( empty $r->previous_id ) {
$pr = $self->revs->get( $r->previous_id );
if ( $pr && $pr->is_placeholder_rev ) {
## Ignore placeholders for the purposes of diffing files
$pr = $self->revs->get( $pr->previous_id );
}
}
if ( $pr ## Can't delta unless we have $pr
&& exists $self->{FILES}->{$pr->id}
&& $encoding eq "none" ## base64, should't delta.
) {
$w->start_delta( type => 'diff-u', encoding => 'none' ) ;
my $old_cp = $self->{FILES}->{$pr->id};
die "no old work path for '", $pr->id, "'\n"
if empty $old_cp ;
die "old work path '$old_cp' not found for '", $pr->id, "'\n"
unless -f $old_cp ;
## TODO: Include entire contents if diff is larger than the contents.
## Accumulate a bunch of output so that characters can make a
## knowledgable CDATA vs <& escaping decision.
my @output ;
my $outlen = 0 ;
my $delete_nl ;
## TODO: Write a "minimal" diff output handler that doesn't
## emit any lines from $old_cp, since they are redundant.
debug "diffing $old_cp $cp" if debugging;
diff "$old_cp", "$cp", ## VCP::RefCountedFile must be stringified
{
## Not passing file names, so no filename header.
STYLE => "VCP::DiffFormat",
OUTPUT => sub {
push @output, $_[0] ;
## Assume no lines split between \r and \n because
## diff() splits based on lines, so we can just
## do a simple conversion here.
$output[-1] =~ s/\r\n|\r/\n/g if $convert_crs ;
$outlen += length $_[0] ;
return unless $outlen > 100_000 ;
_emit_characters( $w, \join "", splice @output ) ;
},
} ;
_emit_characters( $w, \join "", splice @output ) if $outlen ;
$w->end_delta ;
$w->setDataMode( 1 ) unless $self->{NO_INDENT};
}
else {
## Full content, no delta.
$w->start_content( encoding => $encoding ) ;
my $delete_nl ;
while () {
## Odd chunk size is because base64 is most concise with
## chunk sizes a multiple of 57 bytes long.
$read = sysread( F, $buf, 57_000 ) ;
die "$! reading $cp\n" unless defined $read ;
last unless $read ;
if ( $convert_crs ) {
substr( $buf, 0, 1 ) = ""
if $delete_nl && substr( $buf, 0, 1 ) eq "\n" ;
$delete_nl = substr( $buf, -1 ) eq "\n" ;
$buf =~ s/(\r\n|\r)/\n/g ; ## ouch, that's gotta hurt.
}
if ( $encoding eq "none" ) {
_emit_characters( $w, \$buf ) ;
}
else {
$w->characters( encode_base64( $buf ) ) ;
}
}
$w->end_content ;
$w->setDataMode( 1 ) unless $self->{NO_INDENT};
} ;
$digestion = 1 ;
}
if ( $digestion ) {
## TODO: See if this should be seek or sysseek.
sysseek F, 0, 0 or die "$!: $cp" ;
my $d= Digest::MD5->new ;
## gotta do this by hand, since it's in binmode and we want
## to handle ^Z and lone \r's.
my $delete_nl ;
my $read ;
my $buf ;
while () {
$read = sysread( F, $buf, 10_000 ) ;
die "$! reading $cp\n" unless defined $read ;
last unless $read ;
if ( $convert_crs ) {
substr( $buf, 0, 1 ) = ""
if $delete_nl && substr( $buf, 0, 1 ) eq "\n" ;
$delete_nl = substr( $buf, -1 ) eq "\n" ;
$buf =~ s/(\r\n|\r)/\n/g ; ## ouch, that's gotta hurt.
}
$d->add( $buf ) ;
}
$d->addfile( \*F ) ;
$w->digest( $d->b64digest, type => 'MD5', encoding => 'base64' ) ;
}
if ( $close_it ) {
close F ;
}
$w->end_rev ;
1;
} or die "$@ while writing ", defined $r ? $r->as_string : "UNDEF $r!!";
pr_doing;
$self->head_revs->set( [ $r->source_repo_id, $r->source_filebranch_id ],
$r->source_rev_id );
}
sub handle_footer {
my $self = shift ;
my ( $footer ) = @_ ;
$self->writer->endAllTags() if $self->{SEEN_REV};
$self->{SEEN_REV} = 0;
$self->SUPER::handle_footer;
return ;
}
sub writer {
my $self = shift ;
$self->{WRITER} = shift if @_ ;
return $self->{WRITER} ;
}
=back
=head1 LIMITATIONS
Can consume all available memory and disk space because this driver keeps
all old revisions around (metadata in RAM, file images on disk) so that
diff()s may be run against the parent revision when a branch is made.
This behavior is not completely necessary, but because RevML is
currently only used for testing and analysis of small filesets, this
limitation is acceptable.
We can add a command line option to disable this behavior by limiting
(to 0 or to some cache size) the number of revs kept around and letting
files that have no previous rev handy be emitted in their entirety
instead of use a diff. We could also use an on-disk store for the
metadata if that would help. Let us know your needs.
=head1 AUTHOR
Barrie Slaymaker <barries@slaysys.com>
=head1 COPYRIGHT
Copyright (c) 2000, 2001, 2002 Perforce Software, Inc.
All rights reserved.
See L<VCP::License|VCP::License> (C<vcp help license>) for the terms of use.
=cut
1
syntax highlighted by Code2HTML, v. 0.9.1