Artifact Content
Not logged in

Artifact 80c1f4022f70bf8e507ee0e44684fb60c22e55cb

File tools/import-cvs.tcl part of check-in [be32ebcb41] - Redid the logging system aka user feedback completely. Verbosity levels, influenced by the new -v switch. Indentations in the output removed, parsing by tools easier, still human readable. Adapted all users of the previous feedback code to use the new system. by aku on 2007-09-08 05:35:02.

#!/bin/sh
# -*- tcl -*- \
exec tclsh "$0" ${1+"$@"}

# -----------------------------------------------------------------------------

# Import the trunk of a CVS repository wholesale into a fossil repository.

# Limitations implicitly mentioned:
# - No incremental import.
# - No import of branches.

# WIBNI features (beyond eliminating the limitations):
# - Restrict import to specific directory subtrees (SF projects use
#   one repository for several independent modules. Examples: tcllib
#   -> tcllib, tklib, tclapps, etc.). The restriction would allow import
#   of only a specific module.
# - Related to the previous, strip elements from the base path to keep
#   it short.
# - Export to CVS, trunk, possibly branches. I.e. extend the system to be
#   a full bridge. Either Fossil or CVS could be the master repository.

# HACKS. I.e. I do not know if the 'fixes' I use are the correct way
#        of handling the encountered situations.
#
# - File F has archives F,v and Attic/F,v. Currently I will ignore the
#   file in the Attic.
#   Examples: sqlite/os_unix.h
#
# - A specific revision of a file F cannot be checked out (reported
#   error is 'invalid change text'). This indicates a corrupt RCS
#   file, one or more delta are bad. We report but ignore the problem
#   in a best-effort attempt at getting as much history as possible.
#   Examples: tcllib/tklib/modules/tkpiechart/pie.tcl

# -----------------------------------------------------------------------------
# Make private packages accessible.

lappend auto_path [file join [file dirname [info script]] lib]

# -----------------------------------------------------------------------------
# Requirements

package require Tcl 8.4
package require cvs    ; # Frontend, reading from source repository
package require fossil ; # Backend,  writing to destination repository.
package require tools::log

::tools::log::system import

# -----------------------------------------------------------------------------

proc main {} {
    global argv tot nto cvs fossil ntrunk stopat nmax ntfmt nmfmt

    commandline

    cvs::at       $cvs  ; # Define location of CVS repository
    cvs::scan           ; # Gather revision data from the archives
    cvs::csets          ; # Group changes into sets
    cvs::rtree          ; # Build revision tree (trunk only right now).

    set tot 0.0
    set nto 0

    ::tools::log::write 0 import {Begin conversion}
    ::tools::log::write 0 import {Setting up workspaces}

    cvs::workspace ; # cd's to workspace
    fossil::new    ; # Uses cwd as workspace to connect to.

    set ntrunk [cvs::ntrunk] ; set ntfmt %[string length $ntrunk]s
    set nmax   [cvs::ncsets] ; set nmfmt %[string length $nmax]s

    cvs::foreach_cset cset [cvs::root] {
	import $cset
	if {$stopat == $cset} exit
    }
    cvs::wsclear

    ::tools::log::write 0 import "========= [string repeat = 61]"
    ::tools::log::write 0 import "Imported $nto [expr {($nto == 1) ? "changeset" : "changesets"}]"
    ::tools::log::write 0 import "Within [format %.2f $tot] seconds (avg [format %.2f [expr {$tot/$nto}]] seconds/changeset)"

    fossil::destination $fossil

    ::tools::log::write 0 import Ok.
    return
}


# -----------------------------------------------------------------------------

proc commandline {} {
    global argv cvs fossil nosign debugcommit stopat

    set nosign 0
    set debugcommit 0
    set stopat {}
    set verbosity 0

    while {[string match "-*" [set opt [lindex $argv 0]]]} {
	if {$opt eq "--nosign"} {
	    set nosign 1
	    set argv [lrange $argv 1 end]
	    continue
	}
	if {$opt eq "--debugcommit"} {
	    set debugcommit 1
	    set argv [lrange $argv 1 end]
	    continue
	}
	if {$opt eq "--stopat"} {
	    set stopat [lindex $argv 1] 
	    set argv   [lrange $argv 2 end]
	    continue
	}
	if {$opt eq "-v"} {
	    incr verbosity
	    ::tools::log::verbosity $verbosity
	    set argv   [lrange $argv 1 end]
	    continue
	}
	usage
    }
    if {[llength $argv] != 2} usage
    foreach {cvs fossil} $argv break

    if {
	![file exists      $cvs] ||
	![file readable    $cvs] ||
	![file isdirectory $cvs]
    } {
	usage "CVS directory missing, not readable, or not a directory."
    } elseif {[file exists $fossil]} {
	usage "Fossil destination repository exists already."
    }

    fossil::debugcommit $debugcommit
    return
}

proc usage {{text {}}} {
    global argv0
    puts stderr "Usage: $argv0 ?--nosign? cvs-repository fossil-rpeository"
    if {$text eq ""} return
    puts stderr "       $text"
    exit
}

proc import {cset} {
    global tot nto nosign ntrunk stopat ntfmt nmfmt
    ::tools::log::write 0 import "ChangeSet [format $nmfmt $cset] @ [format $ntfmt $nto]/$ntrunk ([format %6.2f [expr {$nto*100.0/$ntrunk}]]%)"

    if {$stopat == $cset} {
	fossil::commit 1 cvs2fossil $nosign \
	    [cvs::wssetup $cset] \
	    ::cvs::wsignore
	::tools::log::write 1 import {%% STOP}
	return
    }

    set usec [lindex [time {
	foreach {uuid ad rm ch} [fossil::commit 0 cvs2fossil $nosign \
				     [cvs::wssetup $cset] \
				     ::cvs::wsignore] break
    } 1] 0]
    cvs::uuid $cset $uuid

    set sec [expr {$usec/1e6}]
    set tot [expr {$tot + $sec}]
    incr nto

    ::tools::log::write 2 import "== $uuid +${ad}-${rm}*${ch}"
    ::tools::log::write 2 import "st in  [format %.2f $sec] sec"

    set avg [expr {$tot/$nto}]
    set max [expr {$ntrunk * $avg}]
    set rem [expr {$max - $tot}]

    ::tools::log::write 3 import "st avg [format %.2f $avg] sec"
    ::tools::log::write 3 import "st run [format %7.2f $tot] sec [format %6.2f [expr {$tot/60}]] min [format %5.2f [expr {$tot/3600}]] hr"
    ::tools::log::write 3 import "st end [format %7.2f $max] sec [format %6.2f [expr {$max/60}]] min [format %5.2f [expr {$max/3600}]] hr"
    ::tools::log::write 3 import "st rem [format %7.2f $rem] sec [format %6.2f [expr {$rem/60}]] min [format %5.2f [expr {$rem/3600}]] hr"
    return
}

# -----------------------------------------------------------------------------

main
exit