#!/usr/bin/perl -w # ==================================================================== # Licensed to the Apache Software Foundation (ASF) under one # or more contributor license agreements. See the NOTICE file # distributed with this work for additional information # regarding copyright ownership. The ASF licenses this file # to you under the Apache License, Version 2.0 (the # "License"); you may not use this file except in compliance # with the License. You may obtain a copy of the License at # # http://www.apache.org/licenses/LICENSE-2.0 # # Unless required by applicable law or agreed to in writing, # software distributed under the License is distributed on an # "AS IS" BASIS, WITHOUT WARRANTIES OR CONDITIONS OF ANY # KIND, either express or implied. See the License for the # specific language governing permissions and limitations # under the License. # ==================================================================== # A script that allows some simple testing of Subversion, in # particular concurrent read, write and read-write access by the 'svn' # client. It can also create working copy trees containing a large # number of files and directories. All repository access is via the # 'svnadmin' and 'svn' commands. # # This script constructs a repository, and populates it with # files. Then it loops making changes to a subset of the files and # committing the tree. Thus when two, or more, instances are run in # parallel there is concurrent read and write access. Sometimes a # commit will fail due to a commit conflict. This is expected, and is # automatically resolved by updating the working copy. # # Each file starts off containing: # A0 # 0 # A1 # 1 # A2 # . # . # A9 # 9 # # The script runs with an ID in the range 0-9, and when it modifies a # file it modifes the line that starts with its ID. Thus scripts with # different IDs will make changes that can be merged automatically. # # The main loop is then: # # step 1: modify a random selection of files # # step 2: optional sleep or wait for RETURN keypress # # step 3: update the working copy automatically merging out-of-date files # # step 4: try to commit, if not successful go to step 3 otherwise go to step 1 # # To allow break-out of potentially infinite loops, the script will # terminate if it detects the presence of a "stop file", the path to # which is specified with the -S option (default ./stop). This allows # the script to be stopped without any danger of interrupting an 'svn' # command, which experiment shows may require Berkeley db_recover to # be used on the repository. # # Running the Script # ================== # # Use three xterms all with shells on the same directory. In the # first xterm run (note, this will remove anything called repostress # in the current directory) # # % stress.pl -c -s1 # # When the message "Committed revision 1." scrolls pass use the second # xterm to run # # % stress.pl -s1 # # Both xterms will modify, update and commit separate working copies to # the same repository. # # Use the third xterm to touch a file 'stop' to cause the scripts to # exit cleanly, i.e. without interrupting an svn command. # # To run a third, fourth, etc. instance of the script use -i # # % stress.pl -s1 -i2 # % stress.pl -s1 -i3 # # Running several instances at once will cause a *lot* of disk # activity. I have run ten instances simultaneously on a Linux tmpfs # (RAM based) filesystem -- watching ten xterms scroll irregularly # can be quite hypnotic! use strict; use IPC::Open3; use Getopt::Std; use File::Find; use File::Path; use File::Spec::Functions; use Cwd; # The name of this script, for error messages. my $stress = 'stress.pl'; # When testing BDB 4.4 and later with DB_RECOVER enabled, the criteria # for a failed update and commit are a bit looser than otherwise. my $dbrecover = undef; # Repository check/create sub init_repo { my ( $repo, $create, $no_sync, $fsfs ) = @_; if ( $create ) { rmtree([$repo]) if -e $repo; my $svnadmin_cmd = "svnadmin create $repo"; $svnadmin_cmd .= " --fs-type bdb" if not $fsfs; $svnadmin_cmd .= " --bdb-txn-nosync" if $no_sync; system( $svnadmin_cmd) and die "$stress: $svnadmin_cmd: failed: $?\n"; open ( CONF, ">>$repo/conf/svnserve.conf") or die "$stress: open svnserve.conf: $!\n"; print CONF "[general]\nanon-access = write\n"; close CONF or die "$stress: close svnserve.conf: $!\n"; } $repo = getcwd . "/$repo" if not file_name_is_absolute $repo; $dbrecover = 1 if -e "$repo/db/__db.register"; print "$stress: BDB automatic database recovery enabled\n" if $dbrecover; return $repo; } # Check-out a working copy sub check_out { my ( $url, $options ) = @_; my $wc_dir = "wcstress.$$"; mkdir "$wc_dir", 0755 or die "$stress: mkdir wcstress.$$: $!\n"; my $svn_cmd = "svn co $url $wc_dir $options"; system( $svn_cmd ) and die "$stress: $svn_cmd: failed: $?\n"; return $wc_dir; } # Print status and update. The update is to do any required merges. sub status_update { my ( $options, $wc_dir, $wait_for_key, $disable_status, $resolve_conflicts ) = @_; my $svn_cmd = "svn st -u $options $wc_dir"; if ( not $disable_status ) { print "Status:\n"; system( $svn_cmd ) and die "$stress: $svn_cmd: failed: $?\n"; } print "Press return to update/commit\n" if $wait_for_key; read STDIN, $wait_for_key, 1 if $wait_for_key; print "Updating:\n"; $svn_cmd = "svn up --non-interactive $options $wc_dir"; # Check for conflicts during the update. If any exist, we resolve them. my $pid = open3(\*UPDATE_WRITE, \*UPDATE_READ, \*UPDATE_ERR_READ, $svn_cmd); my @conflicts = (); while ( ) { print; s/\r*$//; # [Windows compat] Remove trailing \r's if ( /^C (.*)$/ ) { push(@conflicts, ($1)) } } # Print any errors. my $acceptable_error = 0; while ( ) { print; if ($dbrecover) { s/\r*$//; # [Windows compat] Remove trailing \r's $acceptable_error = 1 if ( /^svn:[ ] ( bdb:[ ]PANIC | DB_RUNRECOVERY ) /x ); } } # Close up the streams. close UPDATE_ERR_READ or die "$stress: close UPDATE_ERR_READ: $!\n"; close UPDATE_WRITE or die "$stress: close UPDATE_WRITE: $!\n"; close UPDATE_READ or die "$stress: close UPDATE_READ: $!\n"; # Get commit subprocess exit status die "$stress: waitpid: $!\n" if $pid != waitpid $pid, 0; die "$stress: unexpected update fail: exit status: $?\n" unless $? == 0 or ( $? == 256 and $acceptable_error ); if ($resolve_conflicts) { foreach my $conflict (@conflicts) { $svn_cmd = "svn resolved $conflict"; system( $svn_cmd ) and die "$stress: $svn_cmd: failed: $?\n"; } } } # Print status, update and commit. The update is to do any required # merges. Returns 0 if the commit succeeds and 1 if it fails due to a # conflict. sub status_update_commit { my ( $options, $wc_dir, $wait_for_key, $disable_status, $resolve_conflicts ) = @_; status_update $options, $wc_dir, $wait_for_key, $disable_status, \ $resolve_conflicts; print "Committing:\n"; # Use current time as log message my $now_time = localtime; # [Windows compat] Must use double quotes for the log message. my $svn_cmd = "svn ci $options $wc_dir -m \"$now_time\""; # Need to handle the commit carefully. It could fail for all sorts # of reasons, but errors that indicate a conflict are "acceptable" # while other errors are not. Thus there is a need to check the # return value and parse the error text. my $pid = open3(\*COMMIT_WRITE, \*COMMIT_READ, \*COMMIT_ERR_READ, $svn_cmd); print while ( ); # Look for acceptable errors, ones we expect to occur due to conflicts my $acceptable_error = 0; while ( ) { print; s/\r*$//; # [Windows compat] Remove trailing \r's $acceptable_error = 1 if ( /^svn:[ ] ( .*out[ ]of[ ]date | Conflict[ ]at | Baseline[ ]incorrect | ) /ix ) or ( $dbrecover and ( /^svn:[ ] ( bdb:[ ]PANIC | DB_RUNRECOVERY ) /x )); } close COMMIT_ERR_READ or die "$stress: close COMMIT_ERR_READ: $!\n"; close COMMIT_WRITE or die "$stress: close COMMIT_WRITE: $!\n"; close COMMIT_READ or die "$stress: close COMMIT_READ: $!\n"; # Get commit subprocess exit status die "$stress: waitpid: $!\n" if $pid != waitpid $pid, 0; die "$stress: unexpected commit fail: exit status: $?\n" if ( $? != 0 and $? != 256 ) or ( $? == 256 and $acceptable_error != 1 ); return $? == 256 ? 1 : 0; } # Get a list of all versioned files in the working copy { my @get_list_of_files_helper_array; sub GetListOfFilesHelper { $File::Find::prune = 1 if $File::Find::name =~ m[/.svn]; return if $File::Find::prune or -d; push @get_list_of_files_helper_array, $File::Find::name; } sub GetListOfFiles { my ( $wc_dir ) = @_; @get_list_of_files_helper_array = (); find( \&GetListOfFilesHelper, $wc_dir); return @get_list_of_files_helper_array; } } # Populate a working copy sub populate { my ( $dir, $dir_width, $file_width, $depth, $pad, $props ) = @_; return if not $depth--; for my $nfile ( 1..$file_width ) { my $filename = "$dir/foo$nfile"; open( FOO, ">$filename" ) or die "$stress: open $filename: $!\n"; for my $line ( 0..9 ) { print FOO "A$line\n$line\n" or die "$stress: write to $filename: $!\n"; map { print FOO $_ x 255, "\n"; } ("a", "b", "c", "d") foreach (1..$pad); } print FOO "\$HeadURL: \$\n" or die "$stress: write to $filename: $!\n" if $props; close FOO or die "$stress: close $filename: $!\n"; my $svn_cmd = "svn add $filename"; system( $svn_cmd ) and die "$stress: $svn_cmd: failed: $?\n"; if ( $props ) { $svn_cmd = "svn propset svn:eol-style native $filename"; system( $svn_cmd ) and die "$stress: $svn_cmd: failed: $?\n"; $svn_cmd = "svn propset svn:keywords HeadURL $filename"; system( $svn_cmd ) and die "$stress: $svn_cmd: failed: $?\n"; } } if ( $depth ) { for my $ndir ( 1..$dir_width ) { my $dirname = "$dir/bar$ndir"; my $svn_cmd = "svn mkdir $dirname"; system( $svn_cmd ) and die "$stress: $svn_cmd: failed: $?\n"; populate( "$dirname", $dir_width, $file_width, $depth, $pad, $props ); } } } # Modify a versioned file in the working copy sub ModFile { my ( $filename, $mod_number, $id ) = @_; # Read file into memory replacing the line that starts with our ID open( FOO, "<$filename" ) or die "$stress: open $filename: $!\n"; my @lines = map { s[(^$id.*)][$1,$mod_number]; $_ } ; close FOO or die "$stress: close $filename: $!\n"; # Write the memory back to the file open( FOO, ">$filename" ) or die "$stress: open $filename: $!\n"; print FOO or die "$stress: print $filename: $!\n" foreach @lines; close FOO or die "$stress: close $filename: $!\n"; } sub ParseCommandLine { my %cmd_opts; my $usage = " usage: stress.pl [-cdfhprW] [-i num] [-n num] [-s secs] [-x num] [-o options] [-D num] [-F num] [-N num] [-P num] [-R path] [-S path] [-U url] where -c cause repository creation -d don't make the status calls -f use --fs-type fsfs during repository creation -h show this help information (other options will be ignored) -i the ID (valid IDs are 0 to 9, default is 0 if -c given, 1 otherwise) -n the number of sets of changes to commit -p add svn:eol-style and svn:keywords properties to the files -r perform update-time conflict resolution -s the sleep delay (-1 wait for key, 0 none) -x the number of files to modify in each commit -o options to pass for subversion client -D the number of sub-directories per directory in the tree -F the number of files per directory in the tree -N the depth of the tree -P the number of 10K blocks with which to pad the file -R the path to the repository -S the path to the file whose presence stops this script -U the URL to the repository (file:///<-R path> by default) -W use --bdb-txn-nosync during repository creation "; # defaults $cmd_opts{'D'} = 2; # number of subdirs per dir $cmd_opts{'F'} = 2; # number of files per dir $cmd_opts{'N'} = 2; # depth $cmd_opts{'P'} = 0; # padding blocks $cmd_opts{'R'} = "repostress"; # repository name $cmd_opts{'S'} = "stop"; # path of file to stop the script $cmd_opts{'U'} = "none"; # URL $cmd_opts{'W'} = 0; # create with --bdb-txn-nosync $cmd_opts{'c'} = 0; # create repository $cmd_opts{'d'} = 0; # disable status $cmd_opts{'f'} = 0; # create with --fs-type fsfs $cmd_opts{'h'} = 0; # help $cmd_opts{'i'} = 0; # ID $cmd_opts{'n'} = 200; # sets of changes $cmd_opts{'p'} = 0; # add file properties $cmd_opts{'r'} = 0; # conflict resolution $cmd_opts{'s'} = -1; # sleep interval $cmd_opts{'x'} = 4; # files to modify $cmd_opts{'o'} = ""; # no options passed getopts( 'cdfhi:n:prs:x:o:D:F:N:P:R:S:U:W', \%cmd_opts ) or die $usage; # print help info (and exit nicely) if requested if ( $cmd_opts{'h'} ) { print( $usage ); exit 0; } # default ID if not set $cmd_opts{'i'} = 1 - $cmd_opts{'c'} if not $cmd_opts{'i'}; die $usage if $cmd_opts{'i'} !~ /^[0-9]$/; return %cmd_opts; } ############################################################################ # Main # Why the fixed seed? I use this script for more than stress testing, # I also use it to create test repositories. When creating a test # repository, while I don't care exactly which files get modified, I # find it useful for the repositories to be reproducible, i.e. to have # the same files modified each time. When using this script for # stress testing one could remove this fixed seed and Perl will # automatically use a pseudo-random seed. However it doesn't much # matter, the stress testing really depends on the real-time timing # differences between mutiple instances of the script, rather than the # randomness of the chosen files. srand 123456789; my %cmd_opts = ParseCommandLine(); my $repo = init_repo( $cmd_opts{'R'}, $cmd_opts{'c'}, $cmd_opts{'W'}, $cmd_opts{'f'} ); # [Windows compat] # Replace backslashes in the path, and tweak the number of slashes # in the scheme separator to make the URL always correct. my $urlsep = ($repo =~ m/^\// ? '//' : '///'); $repo =~ s/\\/\//g; # Make URL from path if URL not explicitly specified $cmd_opts{'U'} = "file:$urlsep$repo" if $cmd_opts{'U'} eq "none"; my $wc_dir = check_out $cmd_opts{'U'}, $cmd_opts{'o'}; if ( $cmd_opts{'c'} ) { my $svn_cmd = "svn mkdir $wc_dir/trunk"; system( $svn_cmd ) and die "$stress: $svn_cmd: failed: $?\n"; populate( "$wc_dir/trunk", $cmd_opts{'D'}, $cmd_opts{'F'}, $cmd_opts{'N'}, $cmd_opts{'P'}, $cmd_opts{'p'} ); status_update_commit $cmd_opts{'o'}, $wc_dir, 0, 1 and die "$stress: populate checkin failed\n"; } my @wc_files = GetListOfFiles $wc_dir; die "$stress: not enough files in repository\n" if $#wc_files + 1 < $cmd_opts{'x'}; my $wait_for_key = $cmd_opts{'s'} < 0; my $stop_file = $cmd_opts{'S'}; for my $mod_number ( 1..$cmd_opts{'n'} ) { my @chosen; for ( 1..$cmd_opts{'x'} ) { # Extract random file from list and modify it my $mod_file = splice @wc_files, int rand $#wc_files, 1; ModFile $mod_file, $mod_number, $cmd_opts{'i'}; push @chosen, $mod_file; } # Reinstate list of files, the order doesn't matter push @wc_files, @chosen; if ( $cmd_opts{'x'} > 0 ) { # Loop committing until successful or the stop file is created 1 while not -e $stop_file and status_update_commit $cmd_opts{'o'}, $wc_dir, $wait_for_key, \ $cmd_opts{'d'}, $cmd_opts{'r'}; } else { status_update $cmd_opts{'o'}, $wc_dir, $wait_for_key, $cmd_opts{'d'}, \ $cmd_opts{'r'}; } # Break out of loop, or sleep, if required print( "stop file '$stop_file' detected\n" ), last if -e $stop_file; sleep $cmd_opts{'s'} if $cmd_opts{'s'} > 0; }