#----------------------------------------------------------------------
# Revision control systems support for Eskil.
#
# Copyright (c) 1998-2008, Peter Spjuth
#
# This program is free software; you can redistribute it and/or modify
# it under the terms of the GNU General Public License as published by
# the Free Software Foundation; either version 2 of the License, or
# (at your option) any later version.
#
# This program is distributed in the hope that it will be useful,
# but WITHOUT ANY WARRANTY; without even the implied warranty of
# MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
# GNU General Public License for more details.
#
# You should have received a copy of the GNU General Public License
# along with this program; see the file COPYING. If not, write to
# the Free Software Foundation, Inc., 59 Temple Place - Suite 330,
# Boston, MA 02111-1307, USA.
#
#----------------------------------------------------------------------
# $Revision$
#----------------------------------------------------------------------
##############################################################################
# Revision Control System specific procedures
##############################################################################
# eskil::rev::XXX::detect {file}
#
# Detect if a file is revision controlled under this system.
# If file is empty, check directory for control.
#
# Returns true if controlled or false if not.
# eskil::rev::XXX::ParseRevs {filename revs}
#
# Figure out revision from a list given by user
#
# Returns a list of revisions to display.
#
# Filename may be empty, the rev corresponds to the working tree
# eskil::rev::XXX::get {filename outfile rev}
#
# Get a revision of a file and place it in outfile.
# rev is in any format understood by this system, and
# should be retrieved from ParseRevs
# eskil::rev::XXX::getPatch {revs files {fileListName {}}}
#
# Get a patch of the file tree, between the revisions given.
# revs is in any format understood by this system, and
# should be retrieved from ParseRevs
# If files is non-empty it is a list of files that should be included.
# If fileListName is given, it is a variable name where to place the
# list of files affected by the patch. The list should be cleaned
# through lsort -dictionary -unique.
# NOTE that current directory must be correct before calling.
# eskil::rev::XXX::getChangedFiles {dir revs}
#
# Get a list of files changed between the revisions given.
# revs is in any format understood by this system, and
# should be retrieved from ParseRevs
# eskil::rev::XXX::commitFile {top args}
#
# If implemented, enables the commit feature when comparing edited
# file(s) agains latest check in.
# If no files are given, all edited files are committed.
# eskil::rev::XXX::revertFile {top args}
#
# If implemented, enables the revert feature when comparing edited
# file(s) agains latest check in.
# If no files are given, all edited files are reverted.
# eskil::rev::XXX::viewLog {top filename revs}
#
# If implemented, enables the log feature when comparing revisions.
# View log between displayed versions
# eskil::rev::XXX::mount {dir rev}
#
# If implemented, directory diff can view revisions for this system.
# Mounts a directory revision as a VFS, and returns the mount point
namespace eval eskil::rev::CVS {}
namespace eval eskil::rev::RCS {}
namespace eval eskil::rev::CT {}
namespace eval eskil::rev::GIT {}
namespace eval eskil::rev::FOSSIL {}
namespace eval eskil::rev::SVN {}
namespace eval eskil::rev::HG {}
namespace eval eskil::rev::BZR {}
namespace eval eskil::rev::P4 {}
proc eskil::rev::CVS::detect {file} {
if {$file eq ""} {
set dir [pwd]
} elseif {[file isdirectory $file]} {
set dir $file
} else {
set dir [file dirname $file]
}
if {[file isdirectory [file join $dir CVS]]} {
if {[auto_execok cvs] ne ""} {
return 1
}
}
return 0
}
proc eskil::rev::SVN::detect {file} {
# From SVN 1.7, there is only a .svn at the top of the checkout
if {[SearchUpwardsFromFile $file .svn]} {
if {[auto_execok svn] ne ""} {
return 1
}
}
return 0
}
proc eskil::rev::HG::detect {file} {
if {[SearchUpwardsFromFile $file .hg]} {
if {[auto_execok hg] ne ""} {
return 1
}
}
return 0
}
proc eskil::rev::BZR::detect {file} {
if {[SearchUpwardsFromFile $file .bzr]} {
if {[auto_execok bzr] ne ""} {
return 1
}
}
return 0
}
proc eskil::rev::RCS::detect {file} {
set dir [file dirname $file]
if {[file isdirectory [file join $dir RCS]] || [file exists $file,v]} {
if {[auto_execok rcs] ne ""} {
return 1
}
}
return 0
}
proc eskil::rev::CT::detect {file} {
if {$file eq ""} {
set dir [pwd]
} else {
set dir [file dirname $file]
}
if {[auto_execok cleartool] != ""} {
set old [pwd]
cd $dir
if { ! [catch {exec cleartool pwv -s} view] && $view != "** NONE **"} {
cd $old
return 1
}
cd $old
}
return 0
}
proc eskil::rev::GIT::detect {file} {
if {[SearchUpwardsFromFile $file .git]} {
if {[auto_execok git] ne ""} {
return 1
}
}
return 0
}
proc eskil::rev::FOSSIL::detect {file} {
if {[SearchUpwardsFromFile $file _FOSSIL_ .fslckout .fos]} {
if {[auto_execok fossil] ne ""} {
return 1
}
}
return 0
}
proc eskil::rev::P4::detect {file} {
if {[auto_execok icmp4] != ""} {
if {[catch {exec csh -c "icmp4 have $file"} p4have]} { return 0 }
if {[lindex $p4have 1] eq "-"} { return 1 }
}
return 0
}
# Find the repo top dir given a file/dir reference.
# Return the tail relative to top dir.
# cands is a list of candidates for top marker
proc GetTopDirCand {ref cands dirName tailName} {
upvar 1 $dirName dir $tailName tail
if {[file isdirectory $ref]} {
set dir $ref
set tail ""
} else {
set dir [file dirname $ref]
set tail [file tail $ref]
}
# Locate the top directory
while {[file readable $dir] && [file isdirectory $dir]} {
set found 0
foreach candidate $cands {
if {[file exists [file join $dir $candidate]]} {
set found 1
break
}
}
if {$found} break
set parent [file dirname $dir]
# Make sure to stop if we reach a dead end
if {$parent eq $dir} break
set tail [file join [file tail $dir] $tail]
set dir $parent
}
}
# Find the repo top dir given a file/dir reference.
# Return the tail relative to top dir.
proc eskil::rev::SVN::GetTopDir {ref dirName tailName} {
upvar 1 $dirName dir $tailName tail
GetTopDirCand $ref .svn dir tail
}
# Find the repo top dir given a file/dir reference.
# Return the tail relative to top dir.
proc eskil::rev::GIT::GetTopDir {ref dirName tailName} {
upvar 1 $dirName dir $tailName tail
GetTopDirCand $ref .git dir tail
}
# Find the repo top dir given a file/dir reference.
# Return the tail relative to top dir.
proc eskil::rev::HG::GetTopDir {ref dirName tailName} {
upvar 1 $dirName dir $tailName tail
GetTopDirCand $ref .hg dir tail
}
# Find the repo top dir given a file/dir reference.
# Return the tail relative to top dir.
proc eskil::rev::FOSSIL::GetTopDir {ref dirName tailName} {
upvar 1 $dirName dir $tailName tail
GetTopDirCand $ref ".fos .fslckout _FOSSIL_" dir tail
}
# Get a CVS revision
proc eskil::rev::CVS::get {filename outfile rev} {
set old ""
set dir [file dirname $filename]
if {$dir != "."} {
set old [pwd]
set outfile [file join [pwd] $outfile]
cd $dir
set filename [file tail $filename]
}
set cmd [list exec cvs -z3 update -p]
if {$rev != ""} {
lappend cmd -r $rev
}
lappend cmd [file nativename $filename] > $outfile
if {[catch {eval $cmd} res]} {
if { ! [string match "*Checking out*" $res]} {
tk_messageBox -icon error -title "CVS error" -message $res
}
}
if {$old != ""} {
cd $old
}
}
# Get a CVS patch
proc eskil::rev::CVS::getPatch {revs files {fileListName {}}} {
if {$::Pref(context) > 0} {
set context $::Pref(context)
} else {
set context 5
}
# TODO: support files
set cmd [list exec cvs diff -U $context]
foreach rev $revs {
lappend cmd -r $rev
}
if {[catch {eval $cmd} res]} {
if { ! [string match "*=========*" $res]} {
tk_messageBox -icon error -title "CVS error" -message $res
return ""
}
}
return $res
}
proc eskil::rev::CVS::getChangedFiles {dir revs} {
# Not supported yet
return ""
}
# Get a SVN revision
proc eskil::rev::SVN::get {filename outfile rev} {
set old ""
set dir [file dirname $filename]
if {$dir != "."} {
set old [pwd]
set outfile [file join [pwd] $outfile]
cd $dir
set filename [file tail $filename]
}
set cmd [list exec svn cat]
if {[string match "*://*" $rev]} {
# Full URL
lappend cmd $rev
} else {
if {$rev != ""} {
lappend cmd -r $rev
}
lappend cmd [file nativename $filename]
}
lappend cmd > $outfile
if {[catch {eval $cmd} res]} {
if { ! [string match "*Checking out*" $res]} {
tk_messageBox -icon error -title "SVN error" -message $res
}
}
if {$old != ""} {
cd $old
}
}
# List local changes in a checkout
# This is used to optimise dirdiff in the case of current vs local.
# For SVN a lot of server calls can thus be avoided.
proc eskil::rev::SVN::localChanges {dir} {
set old [pwd]
cd $dir
set info [exec svn status --ignore-externals -q]
cd $old
set changes {}
foreach line [split $info \n] {
set line [string trim $line]
if {[regexp {\S+$} $line file]} {
lappend changes [file join $dir $file]
}
}
return $changes
}
proc eskil::rev::FOSSIL::localChanges {dir} {
set old [pwd]
cd $dir
set info [exec fossil changes]
cd $old
set changes {}
foreach line [split $info \n] {
set line [string trim $line]
if {[regexp {^\S+\s+(\S+)$} $line -> file]} {
lappend changes [file join $dir $file]
}
}
return $changes
}
proc eskil::rev::GIT::localChanges {dir} {
set old [pwd]
cd $dir
set info [exec git status -s --porcelain]
cd $old
set changes {}
foreach line [split $info \n] {
set line [string trim $line]
if {[regexp {^(\S+)\s+(\S+)$} $line -> pre file]} {
lappend changes [file join $dir $file]
}
}
return $changes
}
# Common helper for SVN revisions
proc eskil::rev::SVN::RevsToCmd {revs} {
set cmd {}
set revs2 {}
foreach rev $revs {
# TODO: What happens in strange combinations ?
if {[string match "*://*" $rev]} {
# Full URL
lappend cmd $rev
} else {
lappend revs2 $rev
}
}
if {[llength $revs2] > 0} {
lappend cmd -r [join $revs2 :]
}
return $cmd
}
# Get a SVN patch
proc eskil::rev::SVN::getPatch {revs files {fileListName {}}} {
set cmd [list exec svn diff]
lappend cmd {*}[RevsToCmd $revs]
lappend cmd {*}$files
if {[catch {eval $cmd} res]} {
tk_messageBox -icon error -title "SVN error" -message $res
return ""
}
if {$fileListName ne ""} {
upvar 1 $fileListName fileList
set fileList {}
# SVN will have lines like this to show files:
#Index: dir1/f11
foreach line [lsearch -all -inline -regexp [split $res \n] {^Index: }] {
if {[regexp {Index: (.*)} $line -> fn]} {
lappend fileList $fn
}
}
set fileList [lsort -dictionary -unique $fileList]
}
return $res
}
proc eskil::rev::SVN::getChangedFiles {dir revs} {
# Must call SVN in top dir to get full changeset
GetTopDir $dir top tail
set cmd [list execDir $top svn diff --summarize]
lappend cmd {*}[RevsToCmd $revs]
if {[catch {eval $cmd} res]} {
tk_messageBox -icon error -title "SVN error" -message $res
return ""
}
# Result is one file per line, with an info word before
set files {}
foreach line [split $res \n] {
if {[regexp {^\S+\s+(.*)} $line -> f]} {
lappend files [file join $top $f]
}
}
return $files
}
# Get a HG revision
proc eskil::rev::HG::get {filename outfile rev} {
set old ""
set dir [file dirname $filename]
if {$dir != "."} {
set old [pwd]
set outfile [file join [pwd] $outfile]
cd $dir
set filename [file tail $filename]
}
set cmd [list exec hg cat]
if {$rev != ""} {
lappend cmd -r $rev
}
lappend cmd [file nativename $filename] > $outfile
if {[catch {eval $cmd} res]} {
if {$res ne ""} {
tk_messageBox -icon error -title "HG error" -message $res
}
}
if {$old != ""} {
cd $old
}
}
# Get a HG patch
proc eskil::rev::HG::getPatch {revs files {fileListName {}}} {
set cmd [list exec hg diff]
foreach rev $revs {
lappend cmd -r $rev
}
lappend cmd "--" {*}$files
if {[catch {eval $cmd} res]} {
tk_messageBox -icon error -title "HG error" -message $res
return ""
}
if {$fileListName ne ""} {
upvar 1 $fileListName fileList
set fileList {}
# HG will have lines like this to show files:
#diff -r 533b1d848a1c dir1/f12
#diff -r 0dba7b280f8f -r 2e84355cc419 f1
foreach line [lsearch -all -inline -regexp [split $res \n] {^diff -}] {
if {[regexp {diff (?:-r \w+\s+)*(.*)$} $line -> fn]} {
lappend fileList $fn
}
}
set fileList [lsort -dictionary -unique $fileList]
}
return $res
}
proc eskil::rev::HG::getChangedFiles {dir revs} {
set cmd [list execDir $dir hg diff --stat]
foreach rev $revs {
lappend cmd -r $rev
}
if {[catch {eval $cmd} res]} {
tk_messageBox -icon error -title "HG error" -message $res
return ""
}
# Result is one file per line, with an info word before
GetTopDir $dir top tail
set files {}
foreach line [split $res \n] {
if {[regexp {(.+)\|} $line -> f]} {
set f [string trim $f]
lappend files [file join $top $f]
}
}
return $files
}
# Get a BZR revision
proc eskil::rev::BZR::get {filename outfile rev} {
set old ""
set dir [file dirname $filename]
if {$dir != "."} {
set old [pwd]
set outfile [file join [pwd] $outfile]
cd $dir
set filename [file tail $filename]
}
set cmd [list exec bzr cat]
if {$rev != ""} {
lappend cmd -r $rev
}
lappend cmd [file nativename $filename] > $outfile
if {[catch {eval $cmd} res]} {
if {$res ne ""} {
tk_messageBox -icon error -title "BZR error" -message $res
}
}
if {$old != ""} {
cd $old
}
}
# Get a BZR patch
proc eskil::rev::BZR::getPatch {revs files {fileListName {}}} {
# TODO: support files
set cmd [list exec bzr diff]
if {[llength $revs] == 2} {
lappend cmd -r [lindex $revs 0]..[lindex $revs 1]
} elseif {[llength $revs] == 1} {
lappend cmd -r [lindex $revs 0]
}
if {[catch {eval $cmd} res]} {
if { ! [string match "*===*" $res]} {
tk_messageBox -icon error -title "BZR error" -message $res
return ""
}
}
return $res
}
proc eskil::rev::BZR::getChangedFiles {dir revs} {
# Not supported yet
return ""
}
# Get an RCS revision
proc eskil::rev::RCS::get {filename outfile {rev {}}} {
catch {exec co -p$rev [file nativename $filename] \
> $outfile}
}
# Get a RCS patch
proc eskil::rev::RCS::getPatch {revs files {fileListName {}}} {
# Not supported yet.
return ""
}
proc eskil::rev::RCS::getChangedFiles {dir revs} {
# Not supported yet.
return ""
}
# Get a GIT revision
# No support for revisions yet
proc eskil::rev::GIT::get {filename outfile rev} {
GetTopDir $filename dir tail
if {$rev eq ""} {
set rev HEAD
}
catch {execDir $dir git show $rev:$tail > $outfile}
# example: git show HEAD^^^:apa
}
# Add file to GIT index
proc eskil::rev::GIT::add {filename} {
GetTopDir $filename dir tail
catch {execDir $dir git add $tail}
}
# Get a GIT patch
proc eskil::rev::GIT::getPatch {revs files {fileListName {}}} {
set cmd [list exec git diff -p]
if {[llength $revs] == 0} {
# Always default to HEAD to see changes regardless of index
lappend cmd HEAD
} else {
foreach rev $revs {
lappend cmd $rev
}
}
lappend cmd "--" {*}$files
if {[catch {eval $cmd} res]} {
tk_messageBox -icon error -title "GIT error" -message $res
return ""
}
if {$fileListName ne ""} {
upvar 1 $fileListName fileList
set fileList {}
# GIT will have lines like this to show files:
#diff --git a/dir1/f12 b/dir1/f12
foreach line [lsearch -all -inline -regexp [split $res \n] {^diff -}] {
if {[regexp { a/(.*) b/} $line -> fn]} {
lappend fileList $fn
}
}
set fileList [lsort -dictionary -unique $fileList]
}
return $res
}
# Get a GIT change set
proc eskil::rev::GIT::getChangedFiles {dir revs} {
set cmd [list execDir $dir git diff --name-only]
if {[llength $revs] == 0} {
# Always default to HEAD to see changes regardless of index
lappend cmd HEAD
} else {
foreach rev $revs {
lappend cmd $rev
}
}
if {[catch {eval $cmd} res]} {
tk_messageBox -icon error -title "GIT error" -message $res
return ""
}
# Result is one file per line, relative to repo
GetTopDir $dir top tail
set files {}
foreach line [split $res \n] {
lappend files [file join $top $line]
}
return $files
}
# Get a FOSSIL revision
# No support for revisions yet
proc eskil::rev::FOSSIL::get {filename outfile rev} {
GetTopDir $filename dir tail
if {$rev eq "HEAD" || $rev eq ""} {
catch {execDir $dir fossil finfo -p $tail > $outfile}
} else {
catch {execDir $dir fossil finfo -p $tail -r $rev > $outfile}
}
}
# Get a FOSSIL patch
proc eskil::rev::FOSSIL::getPatch {revs files {fileListName {}}} {
set cmd [list exec fossil diff]
if {[llength $revs] >= 1} {
lappend cmd --from [lindex $revs 0]
}
if {[llength $revs] >= 2} {
lappend cmd --to [lindex $revs 1]
}
# Include added files contents
lappend cmd -N
lappend cmd {*}$files
if {[catch {eval $cmd} res]} {
tk_messageBox -icon error -title "FOSSIL error" -message $res
return ""
}
if {$fileListName ne ""} {
upvar 1 $fileListName fileList
set fileList {}
# FOSSIL will have lines like this to show files:
#Index: dir1/f11
foreach line [lsearch -all -inline -regexp [split $res \n] {^Index: }] {
if {[regexp {Index: (.*)} $line -> fn]} {
lappend fileList $fn
}
}
set fileList [lsort -dictionary -unique $fileList]
}
return $res
}
proc eskil::rev::FOSSIL::getChangedFiles {dir revs} {
set cmd [list execDir $dir fossil diff]
if {[llength $revs] >= 1} {
lappend cmd --from [lindex $revs 0]
}
if {[llength $revs] >= 2} {
lappend cmd --to [lindex $revs 1]
}
lappend cmd --brief
if {[catch {eval $cmd} res]} {
tk_messageBox -icon error -title "FOSSIL error" -message $res
return ""
}
# Result is one file per line, with an info word before
GetTopDir $dir top tail
set files {}
foreach line [split $res \n] {
regexp {\S+\s+(.*)} $line -> f
lappend files [file join $top $f]
}
return $files
}
# Get a ClearCase revision
proc eskil::rev::CT::get {filename outfile rev} {
set filerev [file nativename $filename@@$rev]
if {[catch {exec cleartool get -to $outfile $filerev} msg]} {
tk_messageBox -icon error -title "Cleartool error" -message $msg
return
}
}
# Get a CT patch
proc eskil::rev::CT::getPatch {revs files {fileListName {}}} {
# Not supported yet
return ""
}
proc eskil::rev::CT::getChangedFiles {dir revs} {
# Not supported yet
return ""
}
# Get a P4 revision
proc eskil::rev::P4::get {filename outfile rev} {
set dir [file dirname $filename]
if {[catch {exec csh -c "icmp4 print -q $filename\\#$rev" > $outfile} msg]} {
tk_messageBox -icon error -title "P4 error" -message $msg
return
}
}
# Return current revision of a CVS file
proc eskil::rev::CVS::GetCurrent {filename} {
set old ""
set dir [file dirname $filename]
if {$dir != "."} {
set old [pwd]
cd $dir
set filename [file tail $filename]
}
set cmd [list exec cvs -n status [file nativename $filename]]
if {[catch {eval $cmd} res]} {
# What to do here?
set rev "1.1"
} else {
if { ! [regexp {Working revision:\s+(\d\S*)} $res -> rev]} {
set rev "1.1"
}
}
if {$old != ""} {
cd $old
}
return $rev
}
# Return current revision of a SVN file
proc eskil::rev::SVN::GetCurrent {filename {fullInfo 0}} {
set old ""
if {$filename eq ""} {
set cmd [list exec svn info]
} else {
set dir [file dirname $filename]
if {$dir != "."} {
set old [pwd]
cd $dir
set filename [file tail $filename]
}
set cmd [list exec svn info [file nativename $filename]]
}
if {[catch {eval $cmd} res]} {
# What to do here?
set rev "1"
set res ""
} else {
if { ! [regexp {Last Changed Rev:\s+(\d+)} $res -> rev]} {
set rev "1"
}
}
if {$old != ""} {
cd $old
}
if {$fullInfo} {
return $res
}
return $rev
}
# Return revision list of a SVN file
proc eskil::rev::SVN::GetRevList {filename} {
if {$filename eq ""} {
set cmd [list exec svn log -q -l 50]
} elseif {[string match "*://*" $filename]} {
# Full URL
set cmd [list exec svn log -q -l 50 $filename]
} else {
set cmd [list exec svn log -q -l 50 [file nativename $filename]]
}
if {[catch {eval $cmd} res]} {
# What to do here?
set revs [list 1]
} else {
set lines [lsearch -all -inline -regexp [split $res \n] {^\s*r\d}]
set revs {}
foreach line $lines {
if {[regexp {r(\d+)} $line -> rev]} {
lappend revs $rev
}
}
}
return $revs
}
# Return revision list of a HG file
proc eskil::rev::HG::GetRevList {filename} {
if {$filename eq ""} {
set cmd [list exec hg log -q -l 50]
} else {
set cmd [list exec hg log -q -l 50 [file nativename $filename]]
}
if {[catch {eval $cmd} res]} {
# What to do here?
set revs [list 1]
} else {
set revs {}
foreach line [split $res \n] {
if {[regexp {^(\d+):} $line -> rev]} {
lappend revs $rev
}
}
}
return $revs
}
# Return revision list of a GIT file
proc eskil::rev::GIT::GetRevList {filename} {
set old ""
set cmd [list exec git log --first-parent --oneline -n 50]
if {$filename eq ""} {
# Nothing
} elseif {[file isdirectory $filename]} {
set old [pwd]
cd $filename
} else {
set old [pwd]
cd [file dirname $filename]
lappend cmd [file nativename [file tail $filename]]
}
if {[catch {eval $cmd} res]} {
# What to do here?
puts "ERROR for '$filename' $res"
set revs [list HEAD]
} else {
set lines [split $res \n]
set revs {}
foreach line $lines {
if {[regexp {^(\w+)} $line -> rev]} {
lappend revs $rev
}
}
}
if {$old ne ""} {
cd $old
}
return $revs
}
# Return revision list of a FOSSIL file
proc eskil::rev::FOSSIL::GetRevList {filename} {
# Keep on current branch
set x [execDir $filename fossil branch list]
if { ! [regexp -line {^\* (.*)$} $x -> branch]} {
set branch ""
}
# First, traverse timeline to get a set of ancestor checkins on the
# current branch
set x [execDir $filename fossil timeline ancestors current -t ci -n 5000]
set ancestors {}
set lines ""
set currentArtefact ""
foreach line [split $x \n] {
# Recognise the first line of each checkin
if {[regexp {^\d\d:\d\d:\d\d \[(\w+)\]} $line -> newArtefact]} {
# Check the accumulated lines before this for tags
if {[regexp {tags:\s+([^\)]+)} $lines -> tags]} {
if {$branch eq ""} {
set branch [lindex $tags 0]
}
if {$branch in $tags} {
dict set ancestors $currentArtefact 1
}
}
set currentArtefact $newArtefact
set lines [string trim $line]
} else {
set line [string trim $line]
if {[string index $lines end] eq "-"} {
append lines $line
} else {
append lines \n$line
}
}
}
#puts "Assuming branch '$branch'"
#puts "Found [dict size $ancestors] ancestors in timeline"
if {[file isdirectory $filename]} {
# Just use the ancestors as is. TBD to filter this for a sub directory
return [dict keys $ancestors]
}
# Now get all commits on the file. If finfo had a tag filter,
# this would be much easier.
set x [execDir $filename fossil finfo -l -b $filename]
set fAncestors {}
foreach line [split $x \n] {
if {[regexp {^(\w+)} $line -> artefact]} {
if {[dict exists $ancestors $artefact]} {
lappend fAncestors $artefact
}
}
}
#puts "Found [llength $fAncestors] ancestors for file"
#puts [join $fAncestors \n]
return $fAncestors
}
# Figure out RCS revision from arguments
proc eskil::rev::RCS::ParseRevs {filename revs} {
if {$filename eq ""} {
# RCS does not support tree versions
return {}
}
return $revs
}
# Figure out GIT revision from arguments
# The resulting rev should work with 'git show <rev>:filename'
proc eskil::rev::GIT::ParseRevs {filename revs} {
set result ""
foreach rev $revs {
# Special cases that shortcuts to GIT special names
if {$rev eq "_" || $rev eq "0"} {set rev HEAD}
if {[string is integer -strict $rev] && $rev < 0} {
# A negative integer rev is a relative rev
set revList [eskil::rev::GIT::GetRevList $filename]
set rev [lindex $revList [- $rev]]
if {$rev eq ""} {
set rev [lindex $revs end]
}
}
# Let anything else through
lappend result $rev
}
return $result
}
# Figure out FOSSIL revision from arguments
proc eskil::rev::FOSSIL::ParseRevs {filename revs} {
set result ""
foreach rev $revs {
# Special cases that shortcuts to Fossil special names
if {$rev eq "_" || $rev eq "0"} {set rev current}
# Previous does not work for files
#if {$rev eq "-1"} {set rev previous}
if {[string is integer -strict $rev] && $rev < 0} {
# A negative integer rev is a relative rev
set revList [eskil::rev::FOSSIL::GetRevList $filename]
set rev [lindex $revList [- $rev]]
if {$rev eq ""} {
set rev [lindex $revList end]
}
}
# Let anything else through
lappend result $rev
}
return $result
}
# Figure out HG revision from arguments
proc eskil::rev::HG::ParseRevs {filename revs} {
set result ""
foreach rev $revs {
# Shortcut to HG special names
if {$rev eq "_" || $rev eq "0"} {set rev tip}
if {[string is integer -strict $rev] && $rev < 0} {
# A negative integer rev is a relative rev
set revList [eskil::rev::HG::GetRevList $filename]
set rev [lindex $revList [- $rev]]
if {$rev eq ""} {
set rev [lindex $revList end]
}
}
lappend result $rev
}
return $result
}
# Figure out BZR revision from arguments
proc eskil::rev::BZR::ParseRevs {filename revs} {
set result ""
foreach rev $revs {
# No parsing yet...
lappend result $rev
}
return $result
}
# Figure out CVS revision from arguments
proc eskil::rev::CVS::ParseRevs {filename revs} {
if {$filename eq ""} {
# CVS does not support tree versions
return {}
}
set result {}
foreach rev $revs {
# An integer rev is a relative rev
if {[string is integer -strict $rev]} {
set curr [eskil::rev::CVS::GetCurrent $filename]
regexp {^(.*\.)(\d+)$} $curr -> head tail
set tail [expr {$tail + $rev}]
if {$tail < 1} {set tail 1}
set rev $head$tail
}
lappend result $rev
}
return $result
}
# Look for alternative version in a branch
# Return value, if any, is a full URL to the file
proc eskil::rev::SVN::LookForBranch {filename rev} {
set info [eskil::rev::SVN::GetCurrent $filename 1]
if { ! [regexp -line {URL:\s+(.+)} $info -> URL]} {
return
}
if { ! [regexp -line {Repository Root:\s+(.+)} $info -> Root]} {
return
}
set tail [string range $URL [string length $Root] end]
if { ! [string match "/*" $tail]} {
return
}
set tail [string range $tail 1 end]
set parts [file split $tail]
set alt {}
switch [lindex $parts 0] {
trunk {
lappend alt [file join [lreplace $parts 0 0 branches $rev]]
lappend alt [file join [lreplace $parts 0 0 tags $rev]]
if {$rev eq "trunk"} {
lappend alt [file join [lreplace $parts 0 0 trunk]]
}
}
branches - tags {
if {$rev eq "trunk"} {
lappend alt [file join [lreplace $parts 0 1 trunk]]
}
lappend alt [file join [lreplace $parts 0 1 branches $rev]]
lappend alt [file join [lreplace $parts 0 1 tags $rev]]
}
}
foreach tailAlt $alt {
set urlAlt $Root/[join $tailAlt /]
if {[catch {exec svn "info" $urlAlt} res]} {
continue
}
# Is it enough that svn info worked to check success? Seems so
return $urlAlt
}
return
}
# Figure out SVN revision from arguments
proc eskil::rev::SVN::ParseRevs {filename revs} {
set result {}
foreach rev $revs {
set Url ""
# Non-numeric could be a branch or tag. Look for it.
if { ! [string is integer -strict $rev]} {
if {[regexp {^([^@]+)@(.+)$} $rev -> pre post]} {
set rev $pre
set atRev $post
} else {
set atRev ""
}
set Url [eskil::rev::SVN::LookForBranch $filename $rev]
if {$Url ne ""} {
set rev $atRev
}
}
if {$rev eq "_" || $rev eq "0"} {
# Common names for current
# Use BASE since SVN then knows to use the local copy and avoid
# server calls.
set rev BASE
#set rev [eskil::rev::SVN::GetCurrent $filename]
} elseif {[string is integer -strict $rev] && $rev <= 0} {
# Zero means current
# A negative integer rev is a relative rev
# Get a list from the log
if {$filename eq ""} {
set filename "."
}
if {$Url ne ""} {
set revs [eskil::rev::SVN::GetRevList $Url]
} else {
set revs [eskil::rev::SVN::GetRevList $filename]
}
set rev [lindex $revs [- $rev]]
if {$rev eq ""} {
set rev [lindex $revs end]
}
}
if {$Url ne ""} {
if {$rev ne ""} {
append Url @$rev
}
lappend result $Url
} else {
lappend result $rev
}
}
return $result
}
# Figure out ClearCase revision from arguments
proc eskil::rev::CT::ParseRevs {filename revs} {
if {$filename eq ""} {
# CT does not support tree versions
return {}
}
lassign [eskil::rev::CT::current $filename] stream latest
if {[llength $revs] == 0} {
return [list [file join $stream $latest]]
}
set result {}
foreach rev $revs {
# A negative version number is offset from latest.
set offset 0
set tail [file tail $rev]
if {[string is integer -strict $tail] && $tail < 0} {
set offset $tail
if {$offset == -1} { # Predecessor
return [exec cleartool describe -fmt %PSn $filename]
}
set rev [file dirname $rev]
}
# If the argument is of the form "name/rev", look for a fitting one
if { ! [string is integer $rev] && [regexp {^[^/.]+(/\d+)?$} $rev]} {
if {[catch {exec cleartool lshistory -short $filename} allrevs]} {#
tk_messageBox -icon error -title "Cleartool error" \
-message $allrevs
return
}
set allrevs [split $allrevs \n]
set i [lsearch -glob $allrevs "*$rev" ]
if {$i >= 0} {
set rev [lindex [split [lindex $allrevs $i] "@"] end]
}
}
set rev [file normalize [file join $stream $rev]]
# If we don't have a version number, try to find the latest
if { ! [string is integer [file tail $rev]]} {
if { ! [info exists allrevs]} {
if {[catch {exec cleartool lshistory -short $filename} allrevs]} {#
tk_messageBox -icon error -title "Cleartool error" \
-message $allrevs
return
}
set allrevs [split $allrevs \n]
}
set apa [lsearch -regexp -all -inline $allrevs "$rev/\\d+\$"]
set apa [lindex [lsort -dictionary $apa] end]
if {$apa ne ""} {
set rev [lindex [split $apa "@"] end]
}
}
set tail [file tail $rev]
if {[string is integer -strict $tail] && $offset < 0} {
set path [file dirname $rev]
set tail [expr {$tail + $offset}]
if {$tail < 0} {set tail 0}
set rev [file join $path $tail]
}
lappend result $rev
}
return $result
}
proc eskil::rev::P4::ParseRevs {filename revs} {
if {$revs == ""} { set revs -1 }
foreach rev $revs {
if {[string is digit $rev]} {
lappend result $rev
} else {
if {[catch {exec csh -c "icmp4 files $filename"} res]} {
tk_messageBox -icon error \
-message "Failed p4 files filename: $rev"
exit
}
regexp {\#(\d+)} [file tail $res] -> res
if {$rev != ""} { incr res $rev }
lappend result $res
}
}
return $result
}
# Check in CVS controlled file
proc eskil::rev::CVS::commitFile {top args} {
if {[llength $args] == 0} {
set target all
} elseif {[llength $args] == 1} {
set target [file tail [lindex $args 0]]
} else {
set target "[file tail [lindex $args 0]] ..."
}
set precmd [list cvs -q commit -m]
set postcmd $args
CommitDialog $top $target CVS "" $precmd $postcmd
}
# Check in SVN controlled file
proc eskil::rev::SVN::commitFile {top args} {
if {[llength $args] == 0} {
set target all
} elseif {[llength $args] == 1} {
set target [file tail [lindex $args 0]]
} else {
set target "[file tail [lindex $args 0]] ..."
}
# Any explict dirs mentioned should not recurse.
set precmd [list svn -q commit --depth=empty -m]
set postcmd $args
CommitDialog $top $target SVN "" $precmd $postcmd
}
# Does anything else needs to be committed with this file?
# Typically that would be added directories in SVN.
proc eskil::rev::SVN::commitFileDependency {filename} {
set dir [file dirname $filename]
set result {}
while {$dir ni {. /}} {
set s [exec svn status --depth=empty $dir]
if {[string match "A*" $s]} {
lappend result $dir
} else {
break
}
set dir [file dirname $dir]
}
return $result
}
# Check in HG controlled file
proc eskil::rev::HG::commitFile {top args} {
if {[llength $args] == 0} {
set target all
} elseif {[llength $args] == 1} {
set target [file tail [lindex $args 0]]
} else {
set target "[file tail [lindex $args 0]] ..."
}
set precmd [list hg -q commit -m]
set postcmd $args
CommitDialog $top $target HG "" $precmd $postcmd
}
# Check in GIT controlled file
proc eskil::rev::GIT::commitFile {top args} {
if {[llength $args] == 0} {
set target all
} elseif {[llength $args] == 1} {
set target [file tail [lindex $args 0]]
} else {
set target "[file tail [lindex $args 0]] ..."
}
if {[llength $args] == 0} {
set precmd [list git commit -a -m]
} else {
set precmd [list git commit -m]
}
GetTopDir [pwd] topdir _
set postcmd $args
set gitmsg [CommitDialog $top $target GIT $topdir $precmd $postcmd 1]
if {[string match "*detached HEAD*" $gitmsg]} {
# Make sure to make a detached HEAD commit visible.
tk_messageBox -icon info -title "GIT commit message" -message $gitmsg \
-parent $top
}
}
# Check in Fossil controlled file
proc eskil::rev::FOSSIL::commitFile {top args} {
if {[llength $args] == 0} {
set target all
} elseif {[llength $args] == 1} {
set target [file tail [lindex $args 0]]
} else {
set target "[file tail [lindex $args 0]] ..."
}
set precmd [list fossil commit -m]
set postcmd $args
GetTopDir [pwd] topdir _
# Files to commit might be relative to topdir, take care of that.
# This can happen with -review in a subdir.
set usetopdir ""
foreach f $args {
if { ! [file exists $f]} {
if {[file exists [file join $topdir $f]]} {
set usetopdir $topdir
}
}
}
CommitDialog $top $target Fossil $usetopdir $precmd $postcmd 1
}
# Revert SVN controlled file
proc eskil::rev::SVN::revertFile {top args} {
if {[llength $args] == 0} {
set target all
} elseif {[llength $args] == 1} {
set target [file tail [lindex $args 0]]
} else {
set target "[file tail [lindex $args 0]] ..."
}
set ok [RevertDialog $top $target]
if {$ok ne "ok"} return
if {[llength $args] == 0} {
set args "-R ."
}
set sts [catch {exec svn revert -q {*}$args} svnmsg]
set svnmsg [string trim $svnmsg]
if {$svnmsg ne ""} {
tk_messageBox -icon error -title "SVN revert error" -message $svnmsg \
-parent $top
}
}
# Revert HG controlled file
proc eskil::rev::HG::revertFile {top args} {
if {[llength $args] == 0} {
set target all
} elseif {[llength $args] == 1} {
set target [file tail [lindex $args 0]]
} else {
set target "[file tail [lindex $args 0]] ..."
}
set ok [RevertDialog $top $target]
if {$ok ne "ok"} return
if {[llength $args] == 0} {
set args "--all"
}
set sts [catch {exec hg revert -q -C {*}$args} svnmsg]
set svnmsg [string trim $svnmsg]
if {$svnmsg ne ""} {
tk_messageBox -icon error -title "HG revert error" -message $svnmsg \
-parent $top
}
}
# Revert Fossil controlled file
proc eskil::rev::FOSSIL::revertFile {top args} {
if {[llength $args] == 0} {
set target all
} elseif {[llength $args] == 1} {
set target [file tail [lindex $args 0]]
} else {
set target "[file tail [lindex $args 0]] ..."
}
set ok [RevertDialog $top $target]
if {$ok ne "ok"} return
set sts [catch {exec fossil revert {*}$args} errmsg]
if {$sts} {
tk_messageBox -icon error -title "Fossil revert error" \
-message $errmsg -parent $top
}
}
# Revert Git controlled file
proc eskil::rev::GIT::revertFile {top args} {
if {[llength $args] == 0} {
set target all
} elseif {[llength $args] == 1} {
set target [file tail [lindex $args 0]]
} else {
set target "[file tail [lindex $args 0]] ..."
}
set ok [RevertDialog $top $target]
if {$ok ne "ok"} return
if {[llength $args] == 0} {
set sts [catch {exec git checkout .} gitmsg]
} else {
set sts [catch {exec git checkout {*}$args} gitmsg]
}
set gitmsg [string trim $gitmsg]
if {$sts} {
tk_messageBox -icon error -title "GIT revert error" -message $gitmsg \
-parent $top
}
}
# Mount a directory revision as a VFS, and return the mount point
proc eskil::rev::FOSSIL::mount {dir rev} {
return [vcsvfs::fossil::mount $dir $rev]
}
# Mount a directory revision as a VFS, and return the mount point
proc eskil::rev::SVN::mount {dir rev} {
return [vcsvfs::svn::mount $dir $rev]
}
# Mount a directory revision as a VFS, and return the mount point
proc eskil::rev::HG::mount {dir rev} {
return [vcsvfs::hg::mount $dir $rev]
}
# Mount a directory revision as a VFS, and return the mount point
proc eskil::rev::GIT::mount {dir rev} {
return [vcsvfs::git::mount $dir $rev]
}
# View log between displayed versions
proc eskil::rev::CVS::viewLog {top filename revs} {
set cmd [list exec cvs -q log -N]
if {[llength $revs] > 1} {
lappend cmd -r[join $revs ":"]
} else {
lappend cmd -r[lindex $revs 0]:
}
lappend cmd $filename
if {[catch {eval $cmd} result]} {
#return
}
ViewLog $top $filename $result
}
# View log between displayed versions
proc eskil::rev::SVN::viewLog {top filename revs} {
set cmd [list exec svn log]
if {[llength $revs] > 1} {
lappend cmd -r [join $revs ":"]
} else {
lappend cmd -r HEAD:[lindex $revs 0]
}
lappend cmd $filename
if {[catch {eval $cmd} result]} {
#return
}
ViewLog $top $filename $result
}
# View log between displayed versions
proc eskil::rev::GIT::viewLog {top filename revs} {
set cmd [list exec git log]
if {[llength $revs] > 1} {
lappend cmd [join $revs ".."]
} else {
lappend cmd [lindex $revs 0]..
}
lappend cmd $filename
if {[catch {eval $cmd} result]} {
#return
}
ViewLog $top $filename $result
}
# View log between displayed versions
proc eskil::rev::HG::viewLog {top filename revs} {
set cmd [list exec hg log]
foreach rev $revs {
lappend cmd -r $rev
}
lappend cmd $filename
if {[catch {eval $cmd} result]} {
#return
}
ViewLog $top $filename $result
}
proc eskil::rev::CT::current {filename} {
# Figure out stream and current version
if {[catch {exec cleartool ls $filename} info]} {
tk_messageBox -icon error -title "Cleartool error" -message $info
return
}
set currV {}
if { ! [regexp {@@(\S+)\s+from (\S+)\s+Rule} $info -> dummy currV]} {
regexp {@@(\S+)} $info -> currV
}
set stream [file dirname $currV]
set latest [file tail $currV]
return [list $stream $latest]
}
##############################################################################
# Exported procedures
##############################################################################
# Figure out what revision control system a file is under
# Returns name of rev system if detected, or "" if none.
proc detectRevSystem {file {preference GIT}} {
variable eskil::rev::cache
if {$file ne ""} {
if { ! [file exists $file]} { return "" }
if {[info exists cache($file)]} {
return $cache($file)
}
}
set searchlist [list $preference GIT FOSSIL HG BZR P4]
foreach ns [namespace children eskil::rev] {
lappend searchlist [namespace tail $ns]
}
foreach rev $searchlist {
set result [eskil::rev::${rev}::detect $file]
if {$result} {
set cache($file) $rev
return $rev
}
}
return
}
# Initialise revision control mode
# The file name should be an absolute normalized path.
proc startRevMode {top rev file} {
set ::eskil($top,mode) "rev"
set ::eskil($top,modetype) $rev
set ::eskil($top,rightDir) [file dirname $file]
set ::eskil($top,RevFile) $file
set ::eskil($top,rightLabel) $file
set ::eskil($top,rightFile) $file
set ::eskil($top,rightOK) 1
set ::eskil($top,leftLabel) $rev
set ::eskil($top,leftOK) 0
set ::Pref(toolbar) 1
}
# Prepare for revision diff. Checkout copies of the versions needed.
proc prepareRev {top} {
$::widgets($top,commit) configure -state disabled
$::widgets($top,revert) configure -state disabled
$::widgets($top,log) configure -state disabled
set type $::eskil($top,modetype)
set revs {}
# Search for revision options
if {$::eskil($top,doptrev1) != ""} {
lappend revs $::eskil($top,doptrev1)
}
if {$::eskil($top,doptrev2) != ""} {
lappend revs $::eskil($top,doptrev2)
}
set revs [eskil::rev::${type}::ParseRevs $::eskil($top,RevFile) $revs]
set revlabels {}
foreach rev $revs {
# TODO: In SVN rev could be a full URL, display it nicer
lappend revlabels [GetLastTwoPath $rev]
}
set ::eskil($top,RevRevs) $revs
if {[llength $revs] < 2} {
# Compare local file with specified version.
disallowEdit $top 1
if {[llength $revs] == 0} {
set r ""
set tag "($type)"
} else {
set r [lindex $revs 0]
set tag "($type [lindex $revlabels 0])"
}
set ::eskil($top,leftFile) [tmpFile]
set ::eskil($top,leftLabel) "$::eskil($top,RevFile) $tag"
set ::eskil($top,rightLabel) $::eskil($top,RevFile)
set ::eskil($top,rightFile) $::eskil($top,RevFile)
eskil::rev::${type}::get $::eskil($top,RevFile) $::eskil($top,leftFile) $r
if {[llength $revs] == 0} {
if {[info commands eskil::rev::${type}::commitFile] ne ""} {
$::widgets($top,commit) configure -state normal
}
if {[info commands eskil::rev::${type}::revertFile] ne ""} {
$::widgets($top,revert) configure -state normal
}
}
} else {
# Compare the two specified versions.
disallowEdit $top
set r1 [lindex $revs 0]
set r2 [lindex $revs 1]
set ::eskil($top,leftFile) [tmpFile]
set ::eskil($top,rightFile) [tmpFile]
set ::eskil($top,leftLabel) \
"$::eskil($top,RevFile) ($type [lindex $revlabels 0])"
set ::eskil($top,rightLabel) \
"$::eskil($top,RevFile) ($type [lindex $revlabels 1])"
eskil::rev::${type}::get $::eskil($top,RevFile) $::eskil($top,leftFile) $r1
eskil::rev::${type}::get $::eskil($top,RevFile) $::eskil($top,rightFile) $r2
}
if {[llength $revs] > 0} {
if {[info commands eskil::rev::${type}::viewLog] ne ""} {
$::widgets($top,log) configure -state normal
}
}
# Make sure labels are updated before processing starts
update idletasks
}
# Clean up after a revision diff.
proc cleanupRev {top} {
clearTmp $::eskil($top,rightFile) $::eskil($top,leftFile)
set ::eskil($top,rightFile) $::eskil($top,RevFile)
set ::eskil($top,leftFile) $::eskil($top,RevFile)
}
proc revCommit {top} {
if {[$::widgets($top,commit) cget -state] eq "disabled"} return
set type $::eskil($top,modetype)
if {$::eskil($top,mode) eq "patch"} {
if {[llength $::eskil($top,patchFilelist)] != 0} {
# Use the list extracted from patch
set files $::eskil($top,patchFilelist)
} else {
# Use the list given by user
set files $::eskil($top,reviewFiles)
}
} else {
set files [list $::eskil($top,RevFile)]
}
eskil::rev::${type}::commitFile $top {*}$files
}
proc revRevert {top} {
if {[$::widgets($top,revert) cget -state] eq "disabled"} return
set type $::eskil($top,modetype)
if {$::eskil($top,mode) eq "patch"} {
set files $::eskil($top,reviewFiles)
} else {
set files [list $::eskil($top,RevFile)]
}
eskil::rev::${type}::revertFile $top {*}$files
}
proc revLog {top} {
if {[$::widgets($top,log) cget -state] eq "disabled"} return
set type $::eskil($top,modetype)
eskil::rev::${type}::viewLog $top $::eskil($top,RevFile) \
$::eskil($top,RevRevs)
}
# Get a complete tree patch from this system.
# Note that current directory must be correct before calling.
proc getFullPatch {top} {
$::widgets($top,commit) configure -state disabled
$::widgets($top,revert) configure -state disabled
$::widgets($top,log) configure -state disabled
set type $::eskil($top,modetype)
set files $::eskil($top,reviewFiles)
set revs {}
# Search for revision options
if {$::eskil($top,doptrev1) != ""} {
lappend revs $::eskil($top,doptrev1)
}
if {$::eskil($top,doptrev2) != ""} {
lappend revs $::eskil($top,doptrev2)
}
set revs [eskil::rev::${type}::ParseRevs "" $revs]
set revlabels {}
foreach rev $revs {
lappend revlabels [GetLastTwoPath $rev]
}
if {[llength $revs] == 0} {
if {[info commands eskil::rev::${type}::commitFile] ne ""} {
$::widgets($top,commit) configure -state normal
}
if {[info commands eskil::rev::${type}::revertFile] ne ""} {
$::widgets($top,revert) configure -state normal
}
}
set fileList {}
set patch [eskil::rev::${type}::getPatch $revs $files fileList]
set ::eskil($top,patchFilelist) $fileList
return $patch
}
##############################################################################
# Utilities
##############################################################################
# Execute a command within a specific dir as pwd
proc execDir {dir args} {
set old [pwd]
if {[file isdirectory $dir]} {
cd $dir
} else {
# A file may be given as reference
cd [file dirname $dir]
}
try {
exec {*}$args
} finally {
cd $old
}
}
# Search upwards the directory structure for a file
proc SearchUpwardsFromFile {file args} {
if {$file eq ""} {
set dir [pwd]
} elseif {[file isdirectory $file]} {
set dir $file
} else {
set dir [file dirname $file]
}
while {[file readable $dir] && [file isdirectory $dir]} {
foreach candidate $args {
if {[file exists [file join $dir $candidate]]} {
return 1
}
}
set parent [file dirname $dir]
# Make sure to stop if we reach a dead end
if {$parent eq $dir} break
set dir $parent
}
return 0
}
# Get the last two elements in a file path
proc GetLastTwoPath {path} {
set last [file tail $path]
set penultimate [file tail [file dirname $path]]
if {$penultimate eq "."} {
return $last
} else {
return [file join $penultimate $last]
}
}
# Dialog for commit, getting log message
# target: String shown in dialog
# system: Rev System
# topdir: Directory to execute commit in, if given.
# precmd: Command part before message
# postcmd: Command part after message. Assumed to be files.
# useSts: Use status from exec rather than message to recognise error.
proc CommitDialog {top target system topdir precmd postcmd {useSts 0}} {
set w $top.logmsg
destroy $w
toplevel $w -padx 3 -pady 3
wm title $w "Commit log message for $target"
set ::eskil($top,logdialogok) 0
# Dummy frame used for detecting closed window
ttk::frame $w.dummy -width 10 -height 10
place $w.dummy -x 0 -y 0
text $w.t -width 70 -height 10 -font myfont
if {[info exists ::eskil(logdialog)]} {
$w.t insert end $::eskil(logdialog)
$w.t tag add sel 1.0 end-1c
$w.t mark set insert 1.0
}
ttk::button $w.ok -width 10 -text "Commit" -underline 1 \
-command "set ::eskil($top,logdialogok) 1 ; \
set ::eskil(logdialog) \[$w.t get 1.0 end\] ; \
destroy $w.dummy"
ttk::button $w.ca -width 10 -text "Cancel" -command "destroy $w" \
-underline 0
bind $w <Alt-o> [list $w.ok invoke]\;break
bind $w <Alt-c> [list destroy $w]\;break
bind $w <Key-Escape> [list destroy $w]\;break
grid $w.t - -sticky news -padx 3 -pady 3
grid $w.ok $w.ca -padx 3 -pady 3
grid columnconfigure $w $w.t -weight 1 -uniform a
grid rowconfigure $w $w.t -weight 1
if {[llength $postcmd] > 1} {
# TODO: Scrolled frame maybe? Is dynamic grid enough?
ttk::frame $w.f -padding 1
grid $w.f - -sticky news -padx 3 -pady 3
set t 0
foreach fileName $postcmd {
set ::eskil($top,commit,fileselect$t) 1
ttk::checkbutton $w.f.cb$t -text $fileName \
-variable ::eskil($top,commit,fileselect$t)
incr t
}
dynGridManage $w.f
}
tkwait visibility $w
focus -force $w.t
tkwait window $w.dummy
if { ! $::eskil($top,logdialogok)} {
return
}
set res [string trim $::eskil(logdialog)]
set ::eskil(logdialog) $res
set todo $postcmd
if {[llength $postcmd] > 1} {
# Look through checkbuttons
set todo {}
set t 0
foreach fileName $postcmd {
if {$::eskil($top,commit,fileselect$t)} {
lappend todo $fileName
}
incr t
}
# None left means ignore.
if {[llength $todo] == 0} {
return
}
}
if {[info commands eskil::rev::${system}::commitFileDependency] ne ""} {
foreach filename $todo {
lappend todo {*}[eskil::rev::${system}::commitFileDependency $filename]
}
}
# Splash screen for visual feedback
set now [clock clicks -milliseconds]
ttk::label $w.splash -text "Committing" -anchor center -font myfont
place $w.splash -x 0 -y 0 -relwidth 1.0 -relheight 1.0
update
# Commit
set cmd [list {*}$precmd $res {*}$todo]
if {$topdir ne ""} {
set sts [catch {execDir $topdir {*}$cmd} msg]
} else {
set sts [catch {exec {*}$cmd} msg]
}
set msg [string trim $msg]
if {($useSts && $sts) || (!$useSts && $msg ne "")} {
destroy $w
tk_messageBox -icon error -title "$system commit error" -message $msg \
-parent $top
return
}
# Keep it up for a decent length, regardless of commit delay
while {abs([clock clicks -milliseconds] - $now) < 500} {
after 100
}
destroy $w
return $msg
}
# Dialog for revert acknowledge
proc RevertDialog {top target} {
set msg "Discard local changes for $target ?"
set result [tk_messageBox -type okcancel -icon question -parent $top \
-title "Revert" -message $msg]
return $result
}
# Dialog for log view
proc ViewLog {top filename message} {
set w $top.logview
destroy $w
toplevel $w -padx 3 -pady 3
wm title $w "Log for [file tail $filename]"
text $w.t -width 80 -height 15 -yscrollcommand "$w.sby set" -wrap none
ttk::scrollbar $w.sby -orient vertical -command "$w.t yview"
$w.t insert end $message
ttk::button $w.ok -width 10 -text "Dismiss" -command "destroy $w" \
-underline 0
bind $w <Alt-d> [list destroy $w]\;break
bind $w <Key-Escape> [list destroy $w]\;break
grid $w.t $w.sby -sticky news -padx 3 -pady 3
grid $w.ok - -padx 3 -pady 3
grid columnconfigure $w 0 -weight 1
grid rowconfigure $w 0 -weight 1
}