package VCP::Filter::changesets; =head1 NAME VCP::Filter::changesets - Group revs in to changesets =head1 SYNOPSIS ## From the command line: vcp <source> changesets: ...options... -- <dest> ## In a .vcp file: ChangeSets: time <=60 ## seconds user_id equal ## case-sensitive equality comment equal ## case-sensitive equality branched_rev_branch_id equal ## change only one branch at a time =head1 DESCRIPTION This filter is automatically loaded when there is no sort filter loaded (both this and L<VCP::Filter::sort|VCP::Filter::sort> count as sort filters, more may be added). When all revs from the source have change numbers, this filter sorts by change_id and then by name, regardless of the rules set. The name sort is case sensitive, though it should not be for Win32. If at least one revision arrives from the source with an empty change_id, the rules for this filter establish the conditions that determine what revisions may be grouped in to each change. In this case, this filter rewrites all change_id fields so that the (eventual) destination can use the change_id field to break the revisions in to changes. This is sometimes used by non-changeset oriented destinations to aggregate "changes" as though a user were performing them and to reduce the number of individual operations the destination driver must perform (for instance: VCP::Dest::cvs prefers to not call cvs commit all the time; cvs commit is slow). If you don't specify any conditions, a set of default conditions are used. If you specify any conditions at all, none of the default conditions are used. The default conditions are: time <=60 ## seconds user_id equal ## case-sensitive equality comment equal ## case-sensitive equality branched_rev_branch_id equal ## change only one branch at a time The C<time <=60> condition sets a maximum allowable difference between two revisions; revisions that are more than this number of seconds apart are considered to be in different changes. You may use "equal", "<#" or "<=#" when comparing time and mod_time. If a plain number is provided, <= is assumed. The C<user_id equal> and C<comment equal> conditions assert that two revisions must be by the same user and have the same comment in order to be in the same change. The C<branched_rev_branch_id equal> condition is a special case to handle repositories like CVS which don't record branch creation times. This condition kicks in when a user creates several branches before changing any files on any of them then all of the branches get created at the same time. This condition also kicks in when multiple CVS branches exist with no changes on them. In this case, VCP::Source::cvs groups all of the branch creations after the last "real" edit. The C<branched_rev_branch_id> condition only applies to revisions branching from one branch in to another. For all fields but C<time> and C<mod_time>, you may (for now) only use the C<equal> condition, which is case sensitive equality. An implicit condition is that a parent and a child may not both be altered in the same change. =cut $VERSION = 1 ; use strict ; use VCP::Logger qw( lg pr ); use VCP::Debug qw( :debug ); use VCP::Utils qw( empty ); use VCP::Filter; use VCP::Rev; use base qw( VCP::Filter ); use fields ( 'CMP_SUB', ## Compare two revs for changeset equality 'COMMENT_TIMES', ## The average time of all instances of a comment 'HAS_CHANGE_IDS', ## set if all revs have nonempty change_ids ); sub _eq { defined $_[0] ? defined $_[1] ? $_[0] eq $_[1] : 0 : ! defined $_[1]; } sub _compile_cmp_sub { my VCP::Filter::changesets $self = shift; my ( $rules ) = @_; my @cmps = join "\n && ", map { my ( $field, $cond ) = map lc, @$_; my $cmp; if ( ( $field eq "time" || $field eq "mod_time" ) && $cond =~ /\A(<|<=)?\d+\z/ ) { $cond = "<= $cond" unless defined $1; $cmp = <<CMP; ( empty( \$ra->time ) || empty( \$rb->time ) || ( abs( ( \$ra->time || 0 ) - ( \$rb->time || 0 ) ) $cond ) || ( debugging and debug " times: ", abs( ( \$ra->time || 0 ) - ( \$rb->time || 0 ) ) and 0 ) ) CMP } elsif ( $field eq "branched_rev_branch_id" && $cond eq "equal" ) { $cmp = <<'CMP'; ( ( $ra->is_placeholder_rev ? $rb->is_placeholder_rev ? _eq( $ra->branch_id, $rb->branch_id ) : 0 : ! $rb->is_placeholder_rev ) || ( debugging and debug " is_placeholder_rev or branch_id differs" and 0 ) ) CMP chomp $cmp; } elsif ( $cond eq "equal" ) { $cmp = <<CMP; ( empty( \$ra->$field ) || empty( \$rb->$field ) || _eq( \$ra->$field, \$rb->$field ) || ( debugging and debug " $field differs" and 0 ) ) CMP } else { die "unknown condition in ChangeSets: rule: $field $cond\n"; } $cmp; } @$rules; my @code = ( <<PREAMBLE, " ", @cmps, <<POSTAMBLE ); #line 1 VCP::Filter::changesets::in_same_change() sub { my ( \$ra, \$rb ) = \@_; ## All the empty()s are for VSS (at least) which has very little ## metadata available for delete actions, like user_id and time. debug \$ra->as_string if debugging; debug \$rb->as_string if debugging; my \$in_same_change = PREAMBLE ; debug " in same change" if debugging && \$in_same_change; return \$in_same_change; } POSTAMBLE debug @code if debugging; $self->{CMP_SUB} = eval join "", @code or die "$@ in ChangeSets filter:\n@code"; } sub new { my $class = ref $_[0] ? ref shift : shift; my $self = $class->SUPER::new( @_ ) ; ## Parse the options my ( $spec, $options ) = @_ ; $options ||= []; $self->_compile_cmp_sub( $self->parse_rules_list( $options, "Field", "Condition", [ [qw( time 60 )], [qw( user_id equal )], [qw( comment equal )], [qw( branched_rev_branch_id equal )], ] ) ); return $self ; } sub filter_name { return "ChangeSets" } sub is_sort_filter { 1 } sub handle_header { my VCP::Filter::changesets $self = shift; $self->revs->set; ## clear the list $self->SUPER::handle_header( @_ ); $self->{HAS_CHANGE_IDS} = 1; } sub handle_rev { my VCP::Filter::changesets $self = shift; $self->{HAS_CHANGE_IDS} &&= !empty $_[0]->change_id; $self->revs->add( shift ); } sub _compile_sort_rec_bulk_indexer { my ( $rev, $spec ) = @_ ; my $code = join "", q[#line 1 VCP::Filter::changesets::bulk_indexer], "\n", q[sub { my $revs = shift; my $r; for my $sr ( @$revs ) { $r = $sr->[0]; $sr->[1] = pack( '], map( $rev->pack_format( $_ ), @$spec ), q[', ], map( " " . $rev->index_value_expression( $_ ) . ",\n", @$spec ), q[ ); } }]; debug $code if debugging; return ( eval $code or die $@ ); } sub _calc_sort_recs { my VCP::Filter::changesets $self = shift ; my ( $sort_recs, $spec ) = @_; return unless @$sort_recs; lg "sort key: ", join ", ", map "'$_'", @$spec; my $indexer = _compile_sort_rec_bulk_indexer( $sort_recs->[0]->[0], $spec ); $indexer->( $sort_recs ); } sub sort_revs_by_change_id { my VCP::Filter::changesets $self = shift; ## TODO: see if we can make this more efficient by preindexing all ## these using cmp to compare them. ## TODO: make the name cmd case insensitive if both revisions should ## be compared case insensitively. pr "sorting revisions by change_id, name"; $self->revs->set( sort { VCP::Rev->cmp_id( $a->change_id, $b->change_id ) || $a->name cmp $b->name } $self->revs->get ); } sub sort_revs { my VCP::Filter::changesets $self = shift; ## Use the ->previous references to find the roots and then ## reorder the revs by growing up from the roots. my %rev_kids; my @roots; my @sort_recs; pr "aggregating changes\n"; lg "creating revision trees"; for my $r ( $self->revs->get ) { ## the undef is so the sort keys can be filled in later ## without needing to increase the memory size. my $sort_rec = [ $r, undef ]; push @sort_recs, $sort_rec; if ( $r->previous ) { push @{$rev_kids{int $r->previous}}, $sort_rec; } else { push @roots, $sort_rec; } } lg "generating index"; my @spec = qw( time user_id comment branch_id name ); VCP::Rev::preindex; $self->_calc_sort_recs( \@sort_recs, \@spec ); lg "doing change aggregation"; my @result; @roots = sort { $a->[1] cmp $b->[1] } @roots; my $change_number = 0; my $in_same_change = $self->{CMP_SUB}; while ( @roots ) { ++$change_number; my @change; my @kids; ## Extract one change and then add in all children of the ## extracted revisions. do { my ( $r, undef ) = @{shift @roots}; ## discard sort key push @change, $r; my $kids = delete $rev_kids{int $r}; push @kids, @$kids if $kids; } while ( @roots && $in_same_change->( $change[-1], $roots[0]->[0] ) ); lg "...change $change_number: " . @change . " revs"; ## Do this belatedly in case somebody specifies change_id as an ## differentiation field. $_->change_id( $change_number ) for @change; push @result, @change; if ( @kids ) { ## This is the slow but guaranteed perfect way to sort: #@roots = sort { $a->[1] cmp $b->[1] } @roots, @kids; #next; ## This is the faster but more complex production sorting. ## It's a merge sort with some common cases short circuited ## out. @kids = sort { $a->[1] cmp $b->[1] } @kids if @kids > 1; if ( @roots ) { if ( $kids[0]->[1] ge $roots[-1]->[1] ) { push @roots, @kids; } elsif ( $kids[-1]->[1] le $roots[0]->[1] ) { unshift @roots, @kids; } else { my @result; ## 5 is just a guess. if ( @roots > 5 ) { ## Find the first root that is greater than the first ## kid and splice all preceding roots out. my $i = 0; my $k = $kids[0]->[1]; ++$i while $i <= $#roots && $k ge $roots[$i]->[1]; @result = splice @roots, 0, $i; } ## This is the slowest bit. while ( @roots && @kids ) { my $w = $roots[0]->[1] cmp $kids[0]->[1]; if ( $w < 0 ) { push @result, shift @roots } elsif ( $w > 0 ) { push @result, shift @kids } else { push @result, shift @roots, shift @kids } } @roots = ( @result, @roots, @kids ); } } else { @roots = @kids; } } } pr $change_number, " changes found", $change_number ? sprintf " (%.2f mean revs/change)", $self->revs->get / $change_number : (), "\n"; $self->revs->set( @result ); } sub handle_footer { my VCP::Filter::changesets $self = shift; $self->{HAS_CHANGE_IDS} ? $self->sort_revs_by_change_id : $self->sort_revs; my $revs = $self->revs->remove_all; $self->SUPER::rev_count( 0+@$revs ); for ( @$revs ) { $self->SUPER::handle_rev( $_ ); $_ = undef; } $self->SUPER::handle_footer( @_ ); } sub skip_rev {} sub rev_count {} =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
# | Change | User | Description | Committed | |
---|---|---|---|---|---|
#31 | 5403 | Barrie Slaymaker | - Misc logging, maintainability & debugging improvements | ||
#30 | 4516 | Barrie Slaymaker | - VCP::Filter::changesets supports VCP::Rev::release_ids | ||
#29 | 4507 | Barrie Slaymaker |
- RevML: - added <action>, removed <delete>, <placeholder> and <move> - added <from_id> for clones (and eventually merge actions) - Simplified DTD (can't branch DTD based on which action any more) - VCP::Source::cvs, VCP::Filter::changesets and VCP::Dest::p4 support from_id in <action>clone</action> records - VCP::Dest::perl_data added - VCP::Rev::action() "branch" added, no more undefined action strings - "placeholder" action removed |
||
#28 | 4475 | Barrie Slaymaker |
- my ... if code replaced (thanks, clkao) - exists added (thanks, clkao) |
||
#27 | 4079 | Barrie Slaymaker |
- ChangeSets filter is now 40% or so more memory efficient on large repositories |
||
#26 | 4021 | Barrie Slaymaker |
- Remove all phashes and all base & fields pragmas - Work around SWASHGET error |
||
#25 | 4012 | Barrie Slaymaker | - Remove dependance on pseudohashes (deprecated Perl feature) | ||
#24 | 3996 | Barrie Slaymaker | - Make unknown rev times be ASAP | ||
#23 | 3970 | Barrie Slaymaker |
- VCP::Source handles rev queing, uses disk to reduce RAM - Lots of other fixes |
||
#22 | 3942 | Barrie Slaymaker | - ChangeSets now passes tests | ||
#21 | 3930 | Barrie Slaymaker |
- VCP::Source::cvs and VCP::Dest::p4 handle cloning deletes - "placeholder" actions and is_placeholder_rev() deprecated in favor of is_branch_rev() and is_clone_rev(). - Misc cleanups and minor bugfixes |
||
#20 | 3916 | Barrie Slaymaker | - Reduce memory consumption | ||
#19 | 3911 | Barrie Slaymaker | - ChangeSets no longer deletes a required ARRAY ref | ||
#18 | 3906 | Barrie Slaymaker | - ChangeSets filter now copes with cloned revs | ||
#17 | 3903 | Barrie Slaymaker | - Minor debug logging impovements | ||
#16 | 3902 | Barrie Slaymaker |
- VCP::Filter::changesets does not aggregate creation of multiple branches |
||
#15 | 3855 | Barrie Slaymaker |
- vcp scan, filter, transfer basically functional - Need more work in re: storage format, etc, but functional |
||
#14 | 3850 | Barrie Slaymaker | - No longer stores all revs in memory | ||
#13 | 3830 | Barrie Slaymaker | - VCP::Filter::changesets now ready for disk storage | ||
#12 | 3807 | Barrie Slaymaker | - changesets filter no longer needs VCP::Rev::previous() | ||
#11 | 3769 | Barrie Slaymaker | - avg_comment_time sort key removed | ||
#10 | 3762 | Barrie Slaymaker | - Changeset aggregation now works even when not debugging | ||
#9 | 3737 | Barrie Slaymaker |
- Missing / empty data fields other than branch_id no longer affect changeset aggregation - Needed for VSS, which lacks time information on deletes - If all revs have change_ids, they are now sorted by change_id and name (as opposed to change_id, rev_id) - VCP::Rev::sort_time() removed - VCP::Filter::changesets has better debugging - TestUtils now dumps large diffs |
||
#8 | 3706 | Barrie Slaymaker | - VCP gives some indication of output progress (need more) | ||
#7 | 3702 | Barrie Slaymaker |
- Sort and ChangeSets filters now allow unneeded VCP::Rev objects to be garbage collected. |
||
#6 | 3496 | Barrie Slaymaker | - VSS branching | ||
#5 | 3465 | Barrie Slaymaker | - VCP::Filters can now dump they're sections. | ||
#4 | 3160 | Barrie Slaymaker | Emit "sorting by change_id" to STDERR, too. | ||
#3 | 3155 | Barrie Slaymaker |
Convert to logging using VCP::Logger to reduce stdout/err spew. Simplify & speed up debugging quite a bit. Provide more verbose information in logs. Print to STDERR progress reports to keep users from wondering what's going on. Breaks test; halfway through upgrading run3() to an inline function for speed and for VCP specific features. |
||
#2 | 3127 | Barrie Slaymaker | Minor cleanups. | ||
#1 | 3120 | Barrie Slaymaker | Move changeset aggregation in to its own filter. |