|  | # | 
|  | # Library code for git p4 tests | 
|  | # | 
|  |  | 
|  | # p4 tests never use the top-level repo; always build/clone into | 
|  | # a subdirectory called "$git" | 
|  | TEST_NO_CREATE_REPO=NoThanks | 
|  |  | 
|  | # Some operations require multiple attempts to be successful. Define | 
|  | # here the maximal retry timeout in seconds. | 
|  | RETRY_TIMEOUT=60 | 
|  |  | 
|  | # Sometimes p4d seems to hang. Terminate the p4d process automatically after | 
|  | # the defined timeout in seconds. | 
|  | P4D_TIMEOUT=300 | 
|  |  | 
|  | . ./test-lib.sh | 
|  |  | 
|  | if ! test_have_prereq PYTHON | 
|  | then | 
|  | skip_all='skipping git p4 tests; python not available' | 
|  | test_done | 
|  | fi | 
|  | ( p4 -h && p4d -h ) >/dev/null 2>&1 || { | 
|  | skip_all='skipping git p4 tests; no p4 or p4d' | 
|  | test_done | 
|  | } | 
|  |  | 
|  | # On cygwin, the NT version of Perforce can be used.  When giving | 
|  | # it paths, either on the command-line or in client specifications, | 
|  | # be sure to use the native windows form. | 
|  | # | 
|  | # Older versions of perforce were available compiled natively for | 
|  | # cygwin.  Those do not accept native windows paths, so make sure | 
|  | # not to convert for them. | 
|  | native_path() { | 
|  | path="$1" && | 
|  | if test_have_prereq CYGWIN && ! p4 -V | grep -q CYGWIN | 
|  | then | 
|  | path=$(cygpath --windows "$path") | 
|  | else | 
|  | path=$(test-path-utils real_path "$path") | 
|  | fi && | 
|  | echo "$path" | 
|  | } | 
|  |  | 
|  | # On Solaris the 'date +%s' function is not supported and therefore we | 
|  | # need this replacement. | 
|  | # Attention: This function is not safe again against time offset updates | 
|  | # at runtime (e.g. via NTP). The 'clock_gettime(CLOCK_MONOTONIC)' | 
|  | # function could fix that but it is not in Python until 3.3. | 
|  | time_in_seconds() { | 
|  | python -c 'import time; print int(time.time())' | 
|  | } | 
|  |  | 
|  | # Try to pick a unique port: guess a large number, then hope | 
|  | # no more than one of each test is running. | 
|  | # | 
|  | # This does not handle the case where somebody else is running the | 
|  | # same tests and has chosen the same ports. | 
|  | testid=${this_test#t} | 
|  | git_p4_test_start=9800 | 
|  | P4DPORT=$((10669 + ($testid - $git_p4_test_start))) | 
|  |  | 
|  | P4PORT=localhost:$P4DPORT | 
|  | P4CLIENT=client | 
|  | P4USER=author | 
|  | P4EDITOR=true | 
|  | unset P4CHARSET | 
|  | export P4PORT P4CLIENT P4USER P4EDITOR P4CHARSET | 
|  |  | 
|  | db="$TRASH_DIRECTORY/db" | 
|  | cli="$TRASH_DIRECTORY/cli" | 
|  | git="$TRASH_DIRECTORY/git" | 
|  | pidfile="$TRASH_DIRECTORY/p4d.pid" | 
|  |  | 
|  | # Sometimes "prove" seems to hang on exit because p4d is still running | 
|  | cleanup() { | 
|  | if test -f "$pidfile" | 
|  | then | 
|  | kill -9 $(cat "$pidfile") 2>/dev/null && exit 255 | 
|  | fi | 
|  | } | 
|  | trap cleanup EXIT | 
|  |  | 
|  | # git p4 submit generates a temp file, which will | 
|  | # not get cleaned up if the submission fails.  Don't | 
|  | # clutter up /tmp on the test machine. | 
|  | TMPDIR="$TRASH_DIRECTORY" | 
|  | export TMPDIR | 
|  |  | 
|  | start_p4d() { | 
|  | mkdir -p "$db" "$cli" "$git" && | 
|  | rm -f "$pidfile" && | 
|  | ( | 
|  | cd "$db" && | 
|  | { | 
|  | p4d -q -p $P4DPORT "$@" & | 
|  | echo $! >"$pidfile" | 
|  | } | 
|  | ) && | 
|  |  | 
|  | # This gives p4d a long time to start up, as it can be | 
|  | # quite slow depending on the machine.  Set this environment | 
|  | # variable to something smaller to fail faster in, say, | 
|  | # an automated test setup.  If the p4d process dies, that | 
|  | # will be caught with the "kill -0" check below. | 
|  | i=${P4D_START_PATIENCE:-300} | 
|  | pid=$(cat "$pidfile") | 
|  |  | 
|  | timeout=$(($(time_in_seconds) + $P4D_TIMEOUT)) | 
|  | while true | 
|  | do | 
|  | if test $(time_in_seconds) -gt $timeout | 
|  | then | 
|  | kill -9 $pid | 
|  | exit 1 | 
|  | fi | 
|  | sleep 1 | 
|  | done & | 
|  | watchdog_pid=$! | 
|  |  | 
|  | ready= | 
|  | while test $i -gt 0 | 
|  | do | 
|  | # succeed when p4 client commands start to work | 
|  | if p4 info >/dev/null 2>&1 | 
|  | then | 
|  | ready=true | 
|  | break | 
|  | fi | 
|  | # fail if p4d died | 
|  | kill -0 $pid 2>/dev/null || break | 
|  | echo waiting for p4d to start | 
|  | sleep 1 | 
|  | i=$(( $i - 1 )) | 
|  | done | 
|  |  | 
|  | if test -z "$ready" | 
|  | then | 
|  | # p4d failed to start | 
|  | return 1 | 
|  | fi | 
|  |  | 
|  | # build a p4 user so author@example.com has an entry | 
|  | p4_add_user author | 
|  |  | 
|  | # build a client | 
|  | client_view "//depot/... //client/..." && | 
|  |  | 
|  | return 0 | 
|  | } | 
|  |  | 
|  | p4_add_user() { | 
|  | name=$1 && | 
|  | p4 user -f -i <<-EOF | 
|  | User: $name | 
|  | Email: $name@example.com | 
|  | FullName: Dr. $name | 
|  | EOF | 
|  | } | 
|  |  | 
|  | retry_until_success() { | 
|  | timeout=$(($(time_in_seconds) + $RETRY_TIMEOUT)) | 
|  | until "$@" 2>/dev/null || test $(time_in_seconds) -gt $timeout | 
|  | do | 
|  | sleep 1 | 
|  | done | 
|  | } | 
|  |  | 
|  | retry_until_fail() { | 
|  | timeout=$(($(time_in_seconds) + $RETRY_TIMEOUT)) | 
|  | until ! "$@" 2>/dev/null || test $(time_in_seconds) -gt $timeout | 
|  | do | 
|  | sleep 1 | 
|  | done | 
|  | } | 
|  |  | 
|  | kill_p4d() { | 
|  | pid=$(cat "$pidfile") | 
|  | retry_until_fail kill $pid | 
|  | retry_until_fail kill -9 $pid | 
|  | # complain if it would not die | 
|  | test_must_fail kill $pid >/dev/null 2>&1 && | 
|  | rm -rf "$db" "$cli" "$pidfile" && | 
|  | retry_until_fail kill -9 $watchdog_pid | 
|  | } | 
|  |  | 
|  | cleanup_git() { | 
|  | retry_until_success rm -r "$git" | 
|  | test_must_fail test -d "$git" && | 
|  | retry_until_success mkdir "$git" | 
|  | } | 
|  |  | 
|  | marshal_dump() { | 
|  | what=$1 && | 
|  | line=${2:-1} && | 
|  | cat >"$TRASH_DIRECTORY/marshal-dump.py" <<-EOF && | 
|  | import marshal | 
|  | import sys | 
|  | for i in range($line): | 
|  | d = marshal.load(sys.stdin) | 
|  | print d['$what'] | 
|  | EOF | 
|  | "$PYTHON_PATH" "$TRASH_DIRECTORY/marshal-dump.py" | 
|  | } | 
|  |  | 
|  | # | 
|  | # Construct a client with this list of View lines | 
|  | # | 
|  | client_view() { | 
|  | ( | 
|  | cat <<-EOF && | 
|  | Client: $P4CLIENT | 
|  | Description: $P4CLIENT | 
|  | Root: $cli | 
|  | AltRoots: $(native_path "$cli") | 
|  | LineEnd: unix | 
|  | View: | 
|  | EOF | 
|  | printf "\t%s\n" "$@" | 
|  | ) | p4 client -i | 
|  | } | 
|  |  | 
|  | is_cli_file_writeable() { | 
|  | # cygwin version of p4 does not set read-only attr, | 
|  | # will be marked 444 but -w is true | 
|  | file="$1" && | 
|  | if test_have_prereq CYGWIN && p4 -V | grep -q CYGWIN | 
|  | then | 
|  | stat=$(stat --format=%a "$file") && | 
|  | test $stat = 644 | 
|  | else | 
|  | test -w "$file" | 
|  | fi | 
|  | } |