source: trunk/base/src/macports1.0/macports.tcl @ 64639

Last change on this file since 64639 was 64639, checked in by jmr@…, 11 years ago

fix up running of activate target and autoclean

  • Property svn:eol-style set to native
  • Property svn:keywords set to Id
File size: 118.6 KB
Line 
1# -*- coding: utf-8; mode: tcl; tab-width: 4; indent-tabs-mode: nil; c-basic-offset: 4 -*- vim:fenc=utf-8:filetype=tcl:et:sw=4:ts=4:sts=4
2# macports.tcl
3# $Id: macports.tcl 64639 2010-03-12 15:46:13Z jmr@macports.org $
4#
5# Copyright (c) 2002 Apple Computer, Inc.
6# Copyright (c) 2004 - 2005 Paul Guyot, <pguyot@kallisys.net>.
7# Copyright (c) 2004 - 2006 Ole Guldberg Jensen <olegb@opendarwin.org>.
8# Copyright (c) 2004 - 2005 Robert Shaw <rshaw@opendarwin.org>
9# All rights reserved.
10#
11# Redistribution and use in source and binary forms, with or without
12# modification, are permitted provided that the following conditions
13# are met:
14# 1. Redistributions of source code must retain the above copyright
15#    notice, this list of conditions and the following disclaimer.
16# 2. Redistributions in binary form must reproduce the above copyright
17#    notice, this list of conditions and the following disclaimer in the
18#    documentation and/or other materials provided with the distribution.
19# 3. Neither the name of Apple Computer, Inc. nor the names of its contributors
20#    may be used to endorse or promote products derived from this software
21#    without specific prior written permission.
22#
23# THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS "AS IS"
24# AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE
25# IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE
26# ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT OWNER OR CONTRIBUTORS BE
27# LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR
28# CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF
29# SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS
30# INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN
31# CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE)
32# ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE
33# POSSIBILITY OF SUCH DAMAGE.
34#
35package provide macports 1.0
36package require macports_dlist 1.0
37package require macports_index 1.0
38package require macports_util 1.0
39
40namespace eval macports {
41    namespace export bootstrap_options user_options portinterp_options open_mports ui_priorities port_stages
42    variable bootstrap_options "\
43        portdbpath libpath binpath auto_path extra_env sources_conf prefix portdbformat \
44        portinstalltype portarchivemode portarchivepath portarchivetype portautoclean \
45        porttrace portverbose keeplogs destroot_umask variants_conf rsync_server rsync_options \
46        rsync_dir startupitem_type place_worksymlink xcodeversion xcodebuildcmd \
47        mp_remote_url mp_remote_submit_url configureccache configuredistcc configurepipe buildnicevalue buildmakejobs \
48        applications_dir frameworks_dir developer_dir universal_archs build_arch \
49        macportsuser proxy_override_env proxy_http proxy_https proxy_ftp proxy_rsync proxy_skip"
50    variable user_options "submitter_name submitter_email submitter_key"
51    variable portinterp_options "\
52        portdbpath porturl portpath portbuildpath auto_path prefix prefix_frozen portsharepath \
53        registry.path registry.format registry.installtype portarchivemode portarchivepath \
54        portarchivetype portautoclean porttrace keeplogs portverbose destroot_umask rsync_server \
55        rsync_options rsync_dir startupitem_type place_worksymlink macportsuser \
56        mp_remote_url mp_remote_submit_url configureccache configuredistcc configurepipe buildnicevalue buildmakejobs \
57        applications_dir current_stage frameworks_dir developer_dir universal_archs build_arch $user_options"
58
59    # deferred options are only computed when needed.
60    # they are not exported to the trace thread.
61    # they are not exported to the interpreter in system_options array.
62    variable portinterp_deferred_options "xcodeversion xcodebuildcmd"
63
64    variable open_mports {}
65
66    variable ui_priorities "debug info msg error warn any"
67    variable port_stages "any fetch checksum"
68    variable current_stage "main"
69}
70
71# Provided UI instantiations
72# For standard messages, the following priorities are defined
73#     debug, info, msg, warn, error
74# Clients of the library are expected to provide ui_prefix and ui_channels with
75# the following prototypes.
76#     proc ui_prefix {priority}
77#     proc ui_channels {priority}
78# ui_prefix returns the prefix for the messages, if any.
79# ui_channels returns a list of channels to output the message to, empty for
80#     no message.
81# if these functions are not provided, defaults are used.
82# Clients of the library may optionally provide ui_init with the following
83# prototype.
84#     proc ui_init {priority prefix channels message}
85# ui_init needs to correctly define the proc ::ui_$priority {message} or throw
86# an error.
87# if this function is not provided or throws an error, default procedures for
88# ui_$priority are defined.
89
90# ui_options accessor
91proc macports::ui_isset {val} {
92    if {[info exists macports::ui_options($val)]} {
93        if {$macports::ui_options($val) == "yes"} {
94            return 1
95        }
96    }
97    return 0
98}
99
100
101# global_options accessor
102proc macports::global_option_isset {val} {
103    if {[info exists macports::global_options($val)]} {
104        if {$macports::global_options($val) == "yes"} {
105            return 1
106        }
107    }
108    return 0
109}
110
111proc macports::init_logging {mport} {
112    global macports::channels macports::portdbpath
113
114    if {[getuid] == 0 && [geteuid] != 0} {
115        seteuid 0
116    }
117    if {[catch {macports::ch_logging $mport} err]} {
118        ui_debug "Logging disabled, error opening log file: $err"
119        return 1
120    }
121    # Add our log-channel to all already initialized channels
122    foreach key [array names channels] {
123        set macports::channels($key) [concat $macports::channels($key) "debuglog"]
124    }
125    return 0
126}
127proc macports::ch_logging {mport} {
128    global ::debuglog ::debuglogname
129
130    set portname [_mportkey $mport name]
131    set portpath [_mportkey $mport portpath]
132
133    ui_debug "Starting logging for $portname"
134
135    set logname [macports::getportlogpath $portpath]
136    file mkdir $logname
137    set logname [file join $logname "main.log"]
138
139    set ::debuglogname $logname
140
141    # Truncate the file if already exists
142    set ::debuglog [open $::debuglogname w]
143    puts $::debuglog "version:1"
144}
145proc macports::push_log {mport} {
146    global ::logstack ::logenabled ::debuglog ::debuglogname
147    if {![info exists ::logenabled]} {
148        if {[macports::init_logging $mport] == 0} {
149            set ::logenabled yes
150            set ::logstack [list [list $::debuglog $::debuglogname]]
151            return
152        } else {
153            set ::logenabled no
154        }
155    }
156    if {$::logenabled} {
157        if {[catch {macports::ch_logging $mport} err]} {
158            ui_debug "Logging disabled, error opening log file: $err"
159            return
160        }
161        lappend ::logstack [list $::debuglog $::debuglogname]
162    }
163}
164proc macports::pop_log {} {
165    global ::logenabled ::logstack ::debuglog ::debuglogname
166    if {![info exists ::logenabled]} {
167        return -code error "pop_log called before push_log"
168    }
169    if {$::logenabled && [llength $::logstack] > 0} {
170        close $::debuglog
171        set ::logstack [lreplace $::logstack end end]
172        if {[llength $::logstack] > 0} {
173            set top [lindex $::logstack end]
174            set ::debuglog [lindex $top 0]
175            set ::debuglogname [lindex $top 1]
176        } else {
177            unset ::debuglog
178            unset ::debuglogname
179        }
180    }
181}
182
183proc ui_phase {phase} {
184    global macports::current_stage
185    set macports::current_stage $phase
186    if {$phase != "main"} {
187        set cur_time [clock format [clock seconds] -format  {%+}]
188        ui_debug "$phase phase started at $cur_time"
189    }
190}
191proc ui_message {priority prefix stage args} {
192    global macports::channels ::debuglog macports::current_stage
193    foreach chan $macports::channels($priority) {
194        if {[info exists ::debuglog] && ($chan == "debuglog")} {
195            set chan $::debuglog
196            if {[info exists macports::current_stage]} {
197                set stage $macports::current_stage
198            }
199            set strprefix ":$priority:$stage "
200            if {[lindex $args 0] == "-nonewline"} {
201                puts -nonewline $chan "$strprefix[lindex $args 1]"
202            } else {
203                puts $chan "$strprefix[lindex $args 0]"
204            }
205 
206        } else {
207            if {[lindex $args 0] == "-nonewline"} {
208                puts -nonewline $chan "$prefix[lindex $args 1]"
209            } else {
210                puts $chan "$prefix[lindex $args 0]"
211            }
212        }
213    }
214}
215proc macports::ui_init {priority args} {
216    global macports::channels ::debuglog
217    set default_channel [macports::ui_channels_default $priority]
218    # Get the list of channels.
219    if {[llength [info commands ui_channels]] > 0} {
220        set channels($priority) [ui_channels $priority]
221    } else {
222        set channels($priority) $default_channel
223    }
224   
225    # if some priority initialized after log file is being created
226    if {[info exists ::debuglog]} {
227        set channels($priority) [concat $channels($priority) "debuglog"]
228    }
229    # Simplify ui_$priority.
230    try {
231        set prefix [ui_prefix $priority]
232    } catch * {
233        set prefix [ui_prefix_default $priority]
234    }
235    set stages {fetch checksum}
236    try {
237        eval ::ui_init $priority $prefix $channels($priority) $args
238    } catch * {
239        interp alias {} ui_$priority {} ui_message $priority $prefix ""
240        foreach stage $stages {
241            interp alias {} ui_${priority}_${stage} {} ui_message $priority $prefix $stage
242        }
243    }
244    # Call ui_$priority
245    eval ::ui_$priority $args
246   
247}
248
249# Default implementation of ui_prefix
250proc macports::ui_prefix_default {priority} {
251    switch $priority {
252        debug {
253            return "DEBUG: "
254        }
255        error {
256            return "Error: "
257        }
258        warn {
259            return "Warning: "
260        }
261        default {
262            return ""
263        }
264    }
265}
266
267# Default implementation of ui_channels:
268# ui_options(ports_debug) - If set, output debugging messages
269# ui_options(ports_verbose) - If set, output info messages (ui_info)
270# ui_options(ports_quiet) - If set, don't output "standard messages"
271proc macports::ui_channels_default {priority} {
272    switch $priority {
273        debug {
274            if {[ui_isset ports_debug]} {
275                return {stderr}
276            } else {
277                return {}
278            }
279        }
280        info {
281            if {[ui_isset ports_verbose]} {
282                return {stdout}
283            } else {
284                return {}
285            }
286        }
287        msg {
288            if {[ui_isset ports_quiet]} {
289                return {}
290            } else {
291                return {stdout}
292            }
293        }
294        warn -
295        error {
296            return {stderr}
297        }
298        default {
299            return {stdout}
300        }
301    }
302}
303
304foreach priority ${macports::ui_priorities} {
305    proc ui_$priority {args} [subst { eval macports::ui_init $priority \$args }]
306}
307
308proc ui_warn_once {id msg} {
309    variable macports::warning_done
310    if {![info exists macports::warning_done($id)]} {
311        ui_warn $msg
312        set macports::warning_done($id) 1
313    }
314}
315
316# Replace puts to catch errors (typically broken pipes when being piped to head)
317rename puts tcl::puts
318proc puts {args} {
319    catch "tcl::puts $args"
320}
321
322# find a binary either in a path defined at MacPorts' configuration time
323# or in the PATH environment variable through macports::binaryInPath (fallback)
324proc macports::findBinary {prog {autoconf_hint ""}} {
325    if {${autoconf_hint} != "" && [file executable ${autoconf_hint}]} {
326        return ${autoconf_hint}
327    } else {
328        if {[catch {set cmd_path [macports::binaryInPath ${prog}]} result] == 0} {
329            return ${cmd_path}
330        } else {
331            return -code error "${result} or at its MacPorts configuration time location, did you move it?"
332        }
333    }
334}
335
336# check for a binary in the path
337# returns an error code if it cannot be found
338proc macports::binaryInPath {prog} {
339    global env
340    foreach dir [split $env(PATH) :] {
341        if {[file executable [file join $dir $prog]]} {
342            return [file join $dir $prog]
343        }
344    }
345    return -code error [format [msgcat::mc "Failed to locate '%s' in path: '%s'"] $prog $env(PATH)];
346}
347
348# deferred option processing
349proc macports::getoption {name} {
350    global macports::$name
351    return [expr $$name]
352}
353
354# deferred and on-need extraction of xcodeversion and xcodebuildcmd.
355proc macports::setxcodeinfo {name1 name2 op} {
356    global macports::xcodeversion
357    global macports::xcodebuildcmd
358
359    trace remove variable macports::xcodeversion read macports::setxcodeinfo
360    trace remove variable macports::xcodebuildcmd read macports::setxcodeinfo
361
362    if {[catch {set xcodebuild [binaryInPath "xcodebuild"]}] == 0} {
363        if {![info exists xcodeversion]} {
364            # Determine xcode version (<= 2.0 or 2.1)
365            if {[catch {set xcodebuildversion [exec xcodebuild -version]}] == 0} {
366                if {[regexp "DevToolsCore-(.*); DevToolsSupport-(.*)" $xcodebuildversion devtoolscore_v devtoolssupport_v] == 1} {
367                    if {$devtoolscore_v >= 620.0 && $devtoolssupport_v >= 610.0} {
368                        # for now, we don't need to distinguish 2.1 from 2.1 or higher.
369                        set macports::xcodeversion "2.1"
370                    } else {
371                        set macports::xcodeversion "2.0orlower"
372                    }
373                } else {
374                    set macports::xcodeversion "2.0orlower"
375                }
376            } else {
377                set macports::xcodeversion "2.0orlower"
378            }
379        }
380
381        if {![info exists xcodebuildcmd]} {
382            set macports::xcodebuildcmd "xcodebuild"
383        }
384    } elseif {[catch {set pbxbuild [binaryInPath "pbxbuild"]}] == 0} {
385        if {![info exists xcodeversion]} {
386            set macports::xcodeversion "pb"
387        }
388        if {![info exists xcodebuildcmd]} {
389            set macports::xcodebuildcmd "pbxbuild"
390        }
391    } else {
392        if {![info exists xcodeversion]} {
393            set macports::xcodeversion "none"
394        }
395        if {![info exists xcodebuildcmd]} {
396            set macports::xcodebuildcmd "none"
397        }
398    }
399}
400
401proc mportinit {{up_ui_options {}} {up_options {}} {up_variations {}}} {
402    if {$up_ui_options eq ""} {
403        array set macports::ui_options {}
404    } else {
405        upvar $up_ui_options temp_ui_options
406        array set macports::ui_options [array get temp_ui_options]
407    }
408    if {$up_options eq ""} {
409        array set macports::global_options {}
410    } else {
411        upvar $up_options temp_options
412        array set macports::global_options [array get temp_options]
413    }
414    if {$up_variations eq ""} {
415        array set variations {}
416    } else {
417        upvar $up_variations variations
418    }
419
420    global auto_path env tcl_platform
421    global macports::autoconf::macports_conf_path
422    global macports::macports_user_dir
423    global macports::bootstrap_options
424    global macports::user_options
425    global macports::extra_env
426    global macports::portconf
427    global macports::portdbpath
428    global macports::portsharepath
429    global macports::registry.format
430    global macports::registry.path
431    global macports::sources
432    global macports::sources_default
433    global macports::sources_conf
434    global macports::destroot_umask
435    global macports::libpath
436    global macports::prefix
437    global macports::macportsuser
438    global macports::prefix_frozen
439    global macports::registry.installtype
440    global macports::rsync_dir
441    global macports::rsync_options
442    global macports::rsync_server
443    global macports::variants_conf
444    global macports::xcodebuildcmd
445    global macports::xcodeversion
446    global macports::configureccache
447    global macports::configuredistcc
448    global macports::configurepipe
449    global macports::buildnicevalue
450    global macports::buildmakejobs
451    global macports::universal_archs
452    global macports::build_arch
453
454    # Set the system encoding to utf-8
455    encoding system utf-8
456
457    # Ensure that the macports user directory exists if HOME is defined
458    if {[info exists env(HOME)]} {
459        set macports::macports_user_dir [file normalize $macports::autoconf::macports_user_dir]
460    } else {
461        # Otherwise define the user directory as a direcotory that will never exist
462        set macports::macports_user_dir "/dev/null/NO_HOME_DIR"
463    }
464
465    # Configure the search path for configuration files
466    set conf_files ""
467    lappend conf_files "${macports_conf_path}/macports.conf"
468    if { [file isdirectory $macports_user_dir] } {
469        lappend conf_files "${macports_user_dir}/macports.conf"
470    }
471    if {[info exists env(PORTSRC)]} {
472        set PORTSRC $env(PORTSRC)
473        lappend conf_files ${PORTSRC}
474    }
475
476    # Process the first configuration file we find on conf_files list
477    foreach file $conf_files {
478        if [file exists $file] {
479            set portconf $file
480            set fd [open $file r]
481            while {[gets $fd line] >= 0} {
482                if {[regexp {^(\w+)([ \t]+(.*))?$} $line match option ignore val] == 1} {
483                    if {[lsearch $bootstrap_options $option] >= 0} {
484                        set macports::$option [string trim $val]
485                        global macports::$option
486                    }
487                }
488            }
489            close $fd
490        }
491    }
492
493    # Process per-user only settings
494    set per_user "${macports_user_dir}/user.conf"
495    if [file exists $per_user] {
496        set fd [open $per_user r]
497        while {[gets $fd line] >= 0} {
498            if {[regexp {^(\w+)([ \t]+(.*))?$} $line match option ignore val] == 1} {
499                if {[lsearch $user_options $option] >= 0} {
500                    set macports::$option $val
501                    global macports::$option
502                }
503            }
504        }
505        close $fd
506    }
507
508    if {![info exists sources_conf]} {
509        return -code error "sources_conf must be set in ${macports_conf_path}/macports.conf or in your ${macports_user_dir}/macports.conf file"
510    }
511    set fd [open $sources_conf r]
512    while {[gets $fd line] >= 0} {
513        set line [string trimright $line]
514        if {![regexp {^\s*#|^$} $line]} {
515            if {[regexp {^([\w-]+://\S+)(?:\s+\[(\w+(?:,\w+)*)\])?$} $line _ url flags]} {
516                set flags [split $flags ,]
517                foreach flag $flags {
518                    if {[lsearch -exact [list nosync default] $flag] == -1} {
519                        ui_warn "$sources_conf source '$line' specifies invalid flag '$flag'"
520                    }
521                    if {$flag == "default"} {
522                        if {[info exists sources_default]} {
523                            ui_warn "More than one default port source is defined."
524                        }
525                        set sources_default [concat [list $url] $flags]
526                    }
527                }
528                lappend sources [concat [list $url] $flags]
529            } else {
530                ui_warn "$sources_conf specifies invalid source '$line', ignored."
531            }
532        }
533    }
534    close $fd
535    # Make sure the default port source is defined. Otherwise
536    # [macports::getportresourcepath] fails when the first source doesn't
537    # contain _resources.
538    if {![info exists sources_default]} {
539        ui_warn "No default port source specified in $sources_conf, using last source as default"
540        set sources_default [lindex $sources end]
541    }
542
543    if {![info exists sources]} {
544        if {[file isdirectory ports]} {
545            set sources "file://[pwd]/ports"
546        } else {
547            return -code error "No sources defined in $sources_conf"
548        }
549    }
550
551    if {[info exists variants_conf]} {
552        if {[file exist $variants_conf]} {
553            set fd [open $variants_conf r]
554            while {[gets $fd line] >= 0} {
555                set line [string trimright $line]
556                if {![regexp {^[\ \t]*#.*$|^$} $line]} {
557                    foreach arg [split $line " \t"] {
558                        if {[regexp {^([-+])([-A-Za-z0-9_+\.]+)$} $arg match sign opt] == 1} {
559                            if {![info exists variations($opt)]} {
560                                set variations($opt) $sign
561                            }
562                        } else {
563                            ui_warn "$variants_conf specifies invalid variant syntax '$arg', ignored."
564                        }
565                    }
566                }
567            }
568            close $fd
569        } else {
570            ui_debug "$variants_conf does not exist, variants_conf setting ignored."
571        }
572    }
573    global macports::global_variations
574    array set macports::global_variations [mport_filtervariants [array get variations] yes]
575
576    if {![info exists portdbpath]} {
577        return -code error "portdbpath must be set in ${macports_conf_path}/macports.conf or in your ${macports_user_dir}/macports.conf"
578    }
579    if {![file isdirectory $portdbpath]} {
580        if {![file exists $portdbpath]} {
581            if {[catch {file mkdir $portdbpath} result]} {
582                return -code error "portdbpath $portdbpath does not exist and could not be created: $result"
583            }
584        } else {
585            return -code error "$portdbpath is not a directory. Please create the directory $portdbpath and try again"
586        }
587    }
588
589    set registry.path $portdbpath
590
591    # Format for receipts, can currently be either "flat" or "sqlite"
592    if {[info exists portdbformat]} {
593        if {$portdbformat == "flat" || $portdbformat == "sqlite"} {
594            set registry.format receipt_${portdbformat}
595        } else {
596            return -code error "unknown registry format '$portdbformat' set in macports.conf"
597        }
598    } else {
599        set registry.format receipt_flat
600    }
601
602    # Installation type, whether to use port "images" or install "direct"
603    if {[info exists portinstalltype]} {
604        set registry.installtype $portinstalltype
605    } else {
606        set registry.installtype image
607    }
608
609    # Autoclean mode, whether to automatically call clean after "install"
610    if {![info exists portautoclean]} {
611        set macports::portautoclean "yes"
612        global macports::portautoclean
613    }
614        # whether to keep logs after successful builds
615        if {![info exists keeplogs]} {
616        set macports::keeplogs "no"
617        global macports::keeplogs
618    }
619   
620    # Check command line override for autoclean
621    if {[info exists macports::global_options(ports_autoclean)]} {
622        if {![string equal $macports::global_options(ports_autoclean) $portautoclean]} {
623            set macports::portautoclean $macports::global_options(ports_autoclean)
624        }
625    }
626    # Trace mode, whether to use darwintrace to debug ports.
627    if {![info exists porttrace]} {
628        set macports::porttrace "no"
629        global macports::porttrace
630    }
631    # Check command line override for trace
632    if {[info exists macports::global_options(ports_trace)]} {
633        if {![string equal $macports::global_options(ports_trace) $porttrace]} {
634            set macports::porttrace $macports::global_options(ports_trace)
635        }
636    }
637
638    # Duplicate prefix into prefix_frozen, so that port actions
639    # can always get to the original prefix, even if a portfile overrides prefix
640    set macports::prefix_frozen $prefix
641
642    # Export verbosity.
643    if {![info exists portverbose]} {
644        set macports::portverbose "no"
645        global macports::portverbose
646    }
647    if {[info exists macports::ui_options(ports_verbose)]} {
648        if {![string equal $macports::ui_options(ports_verbose) $portverbose]} {
649            set macports::portverbose $macports::ui_options(ports_verbose)
650        }
651    }
652
653    # Archive mode, whether to create/use binary archive packages
654    if {![info exists portarchivemode]} {
655        set macports::portarchivemode "no"
656        global macports::portarchivemode
657    }
658
659    # Archive path, where to store/retrieve binary archive packages
660    if {![info exists portarchivepath]} {
661        set macports::portarchivepath [file join $portdbpath packages]
662        global macports::portarchivepath
663    }
664    if {$portarchivemode == "yes"} {
665        if {![file isdirectory $portarchivepath]} {
666            if {![file exists $portarchivepath]} {
667                if {![file owned $portdbpath]} {
668                    file lstat $portdbpath stat
669                    return -code error "insufficient privileges for portdbpath $portdbpath (uid $stat(uid)); cannot create portarchivepath"
670                } elseif {[catch {file mkdir $portarchivepath} result]} {
671                    return -code error "portarchivepath $portarchivepath does not exist and could not be created: $result"
672                }
673            }
674        }
675        if {![file isdirectory $portarchivepath]} {
676            return -code error "$portarchivepath is not a directory. Please create the directory $portarchivepath and try again"
677        }
678    }
679
680    # Archive type, what type of binary archive to use (CPIO, gzipped
681    # CPIO, XAR, etc.)
682    if {![info exists portarchivetype]} {
683        set macports::portarchivetype "tgz"
684        global macports::portarchivetype
685    }
686    # Convert archive type to a list for multi-archive support, colon or
687    # comma separators indicates to use multiple archive formats
688    # (reading and writing)
689    set macports::portarchivetype [split $portarchivetype {:,}]
690
691    # Set rync options
692    if {![info exists rsync_server]} {
693        set macports::rsync_server rsync.macports.org
694        global macports::rsync_server
695    }
696    if {![info exists rsync_dir]} {
697        set macports::rsync_dir release/base/
698        global macports::rsync_dir
699    }
700    if {![info exists rsync_options]} {
701        set rsync_options "-rtzv --delete-after"
702        global macports::rsync_options
703    }
704
705    set portsharepath ${prefix}/share/macports
706    if {![file isdirectory $portsharepath]} {
707        return -code error "Data files directory '$portsharepath' must exist"
708    }
709
710    if {![info exists libpath]} {
711        set libpath "${prefix}/share/macports/Tcl"
712    }
713
714    if {![info exists binpath]} {
715        set env(PATH) "${prefix}/bin:${prefix}/sbin:/bin:/sbin:/usr/bin:/usr/sbin"
716    } else {
717        set env(PATH) "$binpath"
718    }
719
720    # Set startupitem default type (can be overridden by portfile)
721    if {![info exists macports::startupitem_type]} {
722        set macports::startupitem_type "default"
723    }
724
725    # Default place_worksymlink
726    if {![info exists macports::place_worksymlink]} {
727        set macports::place_worksymlink yes
728    }
729
730    # Default mp remote options
731    if {![info exists macports::mp_remote_url]} {
732        set macports::mp_remote_url "http://db.macports.org"
733    }
734    if {![info exists macports::mp_remote_submit_url]} {
735        set macports::mp_remote_submit_url "${macports::mp_remote_url}/submit"
736    }
737
738    # Default mp configure options
739    if {![info exists macports::configureccache]} {
740        set macports::configureccache no
741    }
742    if {![info exists macports::configuredistcc]} {
743        set macports::configuredistcc no
744    }
745    if {![info exists macports::configurepipe]} {
746        set macports::configurepipe yes
747    }
748
749    # Default mp build options
750    if {![info exists macports::buildnicevalue]} {
751        set macports::buildnicevalue 0
752    }
753    if {![info exists macports::buildmakejobs]} {
754        set macports::buildmakejobs 0
755    }
756
757    # default user to run as when privileges can be dropped
758    if {![info exists macports::macportsuser]} {
759        set macports::macportsuser $macports::autoconf::macportsuser
760    }
761
762    # Default Xcode Tools path
763    if {![info exists macports::developer_dir]} {
764        set macports::developer_dir "/Developer"
765    }
766
767    # Default mp universal options
768    if {![info exists macports::universal_archs]} {
769        if {[lindex [split $tcl_platform(osVersion) .] 0] >= 10} {
770            set macports::universal_archs {x86_64 i386}
771        } else {
772            set macports::universal_archs {i386 ppc}
773        }
774    }
775   
776    # Default arch to build for
777    if {![info exists macports::build_arch]} {
778        if {$tcl_platform(os) == "Darwin"} {
779            if {[lindex [split $tcl_platform(osVersion) .] 0] >= 10} {
780                if {[sysctl hw.cpu64bit_capable] == 1} {
781                    set macports::build_arch x86_64
782                } else {
783                    set macports::build_arch i386
784                }
785            } else {
786                if {$tcl_platform(machine) == "Power Macintosh"} {
787                    set macports::build_arch ppc
788                } else {
789                    set macports::build_arch i386
790                }
791            }
792        } else {
793            set macports::build_arch ""
794        }
795    } else {
796        set macports::build_arch [lindex $macports::build_arch 0]
797    }
798
799    # ENV cleanup.
800    set keepenvkeys {
801        DISPLAY DYLD_FALLBACK_FRAMEWORK_PATH
802        DYLD_FALLBACK_LIBRARY_PATH DYLD_FRAMEWORK_PATH
803        DYLD_LIBRARY_PATH DYLD_INSERT_LIBRARIES
804        HOME JAVA_HOME MASTER_SITE_LOCAL
805        PATCH_SITE_LOCAL PATH PORTSRC RSYNC_PROXY TMP TMPDIR
806        USER GROUP
807        http_proxy HTTPS_PROXY FTP_PROXY ALL_PROXY NO_PROXY
808        COLUMNS LINES
809    }
810    if {[info exists extra_env]} {
811        set keepenvkeys [concat ${keepenvkeys} ${extra_env}]
812    }
813
814    foreach envkey [array names env] {
815        if {[lsearch $keepenvkeys $envkey] == -1} {
816            array unset env $envkey
817        }
818    }
819
820    if {![info exists xcodeversion] || ![info exists xcodebuildcmd]} {
821        # We'll resolve these later (if needed)
822        trace add variable macports::xcodeversion read macports::setxcodeinfo
823        trace add variable macports::xcodebuildcmd read macports::setxcodeinfo
824    }
825
826    # Set the default umask
827    if {![info exists destroot_umask]} {
828        set destroot_umask 022
829    }
830
831    if {[info exists master_site_local] && ![info exists env(MASTER_SITE_LOCAL)]} {
832        set env(MASTER_SITE_LOCAL) "$master_site_local"
833    }
834
835    if {[file isdirectory $libpath]} {
836        lappend auto_path $libpath
837        set macports::auto_path $auto_path
838
839        # XXX: not sure if this the best place, but it needs to happen
840        # early, and after auto_path has been set.  Or maybe Pextlib
841        # should ship with macports1.0 API?
842        package require Pextlib 1.0
843        package require registry 1.0
844        package require registry2 2.0
845    } else {
846        return -code error "Library directory '$libpath' must exist"
847    }
848
849    # unset environment an extra time, to work around bugs in Leopard Tcl
850    foreach envkey [array names env] {
851        if {[lsearch $keepenvkeys $envkey] == -1} {
852            unsetenv $envkey
853        }
854    }
855
856    # Proxy handling (done this late since Pextlib is needed)
857    if {![info exists proxy_override_env] } {
858        set proxy_override_env "no"
859    }
860    if {[catch {array set sysConfProxies [get_systemconfiguration_proxies]} result]} {
861        return -code error "Unable to get proxy configuration from system: $result"
862    }
863    if {![info exists env(http_proxy)] || $proxy_override_env == "yes" } {
864        if {[info exists proxy_http]} {
865            set env(http_proxy) $proxy_http
866        } elseif {[info exists sysConfProxies(proxy_http)]} {
867            set env(http_proxy) $sysConfProxies(proxy_http)
868        }
869    }
870    if {![info exists env(HTTPS_PROXY)] || $proxy_override_env == "yes" } {
871        if {[info exists proxy_https]} {
872            set env(HTTPS_PROXY) $proxy_https
873        } elseif {[info exists sysConfProxies(proxy_https)]} {
874            set env(HTTPS_PROXY) $sysConfProxies(proxy_https)
875        }
876    }
877    if {![info exists env(FTP_PROXY)] || $proxy_override_env == "yes" } {
878        if {[info exists proxy_ftp]} {
879            set env(FTP_PROXY) $proxy_ftp
880        } elseif {[info exists sysConfProxies(proxy_ftp)]} {
881            set env(FTP_PROXY) $sysConfProxies(proxy_ftp)
882        }
883    }
884    if {![info exists env(RSYNC_PROXY)] || $proxy_override_env == "yes" } {
885        if {[info exists proxy_rsync]} {
886            set env(RSYNC_PROXY) $proxy_rsync
887        }
888    }
889    if {![info exists env(NO_PROXY)] || $proxy_override_env == "yes" } {
890        if {[info exists proxy_skip]} {
891            set env(NO_PROXY) $proxy_skip
892        } elseif {[info exists sysConfProxies(proxy_skip)]} {
893            set env(NO_PROXY) $sysConfProxies(proxy_skip)
894        }
895    }
896
897    # load the quick index
898    _mports_load_quickindex
899
900    set default_source_url [lindex ${sources_default} 0]
901    if {[macports::getprotocol $default_source_url] == "file" || [macports::getprotocol $default_source_url] == "rsync"} {
902        set default_portindex [macports::getindex $default_source_url]
903        if {[file exists $default_portindex] && [expr [clock seconds] - [file mtime $default_portindex]] > 1209600} {
904            ui_warn "port definitions are more than two weeks old, consider using selfupdate"
905        }
906    }
907   
908    # init registry if needed
909    if {${registry.format} == "receipt_sqlite"} {
910        set db_path [file join ${registry.path} registry registry.db]
911        set db_exists [file exists $db_path]
912        registry::open $db_path
913        # for the benefit of the portimage code that is called from multiple interpreters
914        global registry_open
915        set registry_open yes
916        # convert any flat receipts if we just created a new db
917        if {$db_exists == 0 && [file writable $db_path]} {
918            registry::convert_to_sqlite
919        }
920    }
921}
922
923proc macports::worker_init {workername portpath porturl portbuildpath options variations} {
924    global macports::portinterp_options macports::portinterp_deferred_options registry.installtype
925
926    # Hide any Tcl commands that should be inaccessible to port1.0 and Portfiles
927    # exit: It should not be possible to exit the interpreter
928    interp hide $workername exit
929
930    # cd: This is necessary for some code in port1.0, but should be hidden
931    interp eval $workername "rename cd _cd"
932
933    # Tell the sub interpreter about all the Tcl packages we already
934    # know about so it won't glob for packages.
935    foreach pkgName [package names] {
936        foreach pkgVers [package versions $pkgName] {
937            set pkgLoadScript [package ifneeded $pkgName $pkgVers]
938            $workername eval "package ifneeded $pkgName $pkgVers {$pkgLoadScript}"
939        }
940    }
941
942    # Create package require abstraction procedure
943    $workername eval "proc PortSystem \{version\} \{ \n\
944            package require port \$version \}"
945
946    # Clearly separate slave interpreters and the master interpreter.
947    $workername alias mport_exec mportexec
948    $workername alias mport_open mportopen
949    $workername alias mport_close mportclose
950    $workername alias mport_lookup mportlookup
951    $workername alias ui_phase ui_phase
952
953    # instantiate the UI call-backs
954    foreach priority ${macports::ui_priorities} {
955        $workername alias ui_$priority ui_$priority
956        foreach stage ${macports::port_stages} {
957            $workername alias ui_${priority}_${stage} ui_${priority}_${stage}
958        }
959 
960    }
961
962    $workername alias ui_prefix ui_prefix
963    $workername alias ui_channels ui_channels
964   
965    $workername alias ui_warn_once ui_warn_once
966
967    # Export some utility functions defined here.
968    $workername alias macports_create_thread macports::create_thread
969    $workername alias getportworkpath_from_buildpath macports::getportworkpath_from_buildpath
970    $workername alias getportresourcepath macports::getportresourcepath
971    $workername alias getportlogpath macports::getportlogpath
972    $workername alias getdefaultportresourcepath macports::getdefaultportresourcepath
973    $workername alias getprotocol macports::getprotocol
974    $workername alias getportdir macports::getportdir
975    $workername alias findBinary macports::findBinary
976    $workername alias binaryInPath macports::binaryInPath
977    $workername alias sysctl sysctl
978    $workername alias realpath realpath
979    $workername alias _mportsearchpath _mportsearchpath
980    $workername alias _portnameactive _portnameactive
981
982    # New Registry/Receipts stuff
983    $workername alias registry_new registry::new_entry
984    $workername alias registry_open registry::open_entry
985    $workername alias registry_write registry::write_entry
986    $workername alias registry_prop_store registry::property_store
987    $workername alias registry_prop_retr registry::property_retrieve
988    $workername alias registry_exists registry::entry_exists
989    $workername alias registry_exists_for_name registry::entry_exists_for_name
990    $workername alias registry_activate portimage::activate
991    $workername alias registry_deactivate portimage::deactivate
992    $workername alias registry_uninstall portuninstall::uninstall
993    $workername alias registry_register_deps registry::register_dependencies
994    $workername alias registry_fileinfo_for_index registry::fileinfo_for_index
995    $workername alias registry_bulk_register_files registry::register_bulk_files
996    $workername alias registry_active registry::active
997    $workername alias registry_file_registered registry::file_registered
998
999    # deferred options processing.
1000    $workername alias getoption macports::getoption
1001
1002    foreach opt $portinterp_options {
1003        if {![info exists $opt]} {
1004            global macports::$opt
1005        }
1006        if {[info exists $opt]} {
1007            $workername eval set system_options($opt) \{[set $opt]\}
1008            $workername eval set $opt \{[set $opt]\}
1009        }
1010    }
1011
1012    foreach opt $portinterp_deferred_options {
1013        global macports::$opt
1014        # define the trace hook.
1015        $workername eval \
1016            "proc trace_$opt {name1 name2 op} { \n\
1017                trace remove variable ::$opt read ::trace_$opt \n\
1018                global $opt \n\
1019                set $opt \[getoption $opt\] \n\
1020            }"
1021        # next access will actually define the variable.
1022        $workername eval "trace add variable ::$opt read ::trace_$opt"
1023        # define some value now
1024        $workername eval set $opt "?"
1025    }
1026
1027    foreach {opt val} $options {
1028        $workername eval set user_options($opt) $val
1029        $workername eval set $opt $val
1030    }
1031
1032    foreach {var val} $variations {
1033        $workername eval set variations($var) $val
1034    }
1035
1036    if { [info exists registry.installtype] } {
1037        $workername eval set installtype ${registry.installtype}
1038    }
1039}
1040
1041# Create a thread with most configuration options set.
1042# The newly created thread is sent portinterp_options vars and knows where to
1043# find all packages we know.
1044proc macports::create_thread {} {
1045    package require Thread
1046
1047    global macports::portinterp_options
1048
1049    # Create the thread.
1050    set result [thread::create -preserved {thread::wait}]
1051
1052    # Tell the thread about all the Tcl packages we already
1053    # know about so it won't glob for packages.
1054    foreach pkgName [package names] {
1055        foreach pkgVers [package versions $pkgName] {
1056            set pkgLoadScript [package ifneeded $pkgName $pkgVers]
1057            thread::send -async $result "package ifneeded $pkgName $pkgVers {$pkgLoadScript}"
1058        }
1059    }
1060
1061    # inherit configuration variables.
1062    thread::send -async $result "namespace eval macports {}"
1063    foreach opt $portinterp_options {
1064        if {![info exists $opt]} {
1065            global macports::$opt
1066        }
1067        if {[info exists $opt]} {
1068            thread::send -async $result "global macports::$opt"
1069            set val [set macports::$opt]
1070            thread::send -async $result "set macports::$opt \"$val\""
1071        }
1072    }
1073
1074    return $result
1075}
1076
1077proc macports::fetch_port {url} {
1078    global macports::portdbpath tcl_platform
1079    set fetchdir [file join $portdbpath portdirs]
1080    set fetchfile [file tail $url]
1081    file mkdir $fetchdir
1082    if {![file writable $fetchdir]} {
1083        return -code error "Port remote fetch failed: You do not have permission to write to $fetchdir"
1084    }
1085    if {[catch {curl fetch $url [file join $fetchdir $fetchfile]} result]} {
1086        return -code error "Port remote fetch failed: $result"
1087    }
1088    cd $fetchdir
1089    if {[catch {exec [findBinary tar $macports::autoconf::tar_path] -zxf $fetchfile} result]} {
1090        return -code error "Port extract failed: $result"
1091    }
1092    if {[regexp {(.+).tgz} $fetchfile match portdir] != 1} {
1093        return -code error "Can't decipher portdir from $fetchfile"
1094    }
1095    return [file join $fetchdir $portdir]
1096}
1097
1098proc macports::getprotocol {url} {
1099    if {[regexp {(?x)([^:]+)://.+} $url match protocol] == 1} {
1100        return ${protocol}
1101    } else {
1102        return -code error "Can't parse url $url"
1103    }
1104}
1105
1106# XXX: this really needs to be rethought in light of the remote index
1107# I've added the destdir parameter.  This is the location a remotely
1108# fetched port will be downloaded to (currently only applies to
1109# mports:// sources).
1110proc macports::getportdir {url {destdir "."}} {
1111    set protocol [macports::getprotocol $url]
1112    switch ${protocol} {
1113        file {
1114            return [file normalize [string range $url [expr [string length $protocol] + 3] end]]
1115        }
1116        mports {
1117            return [macports::index::fetch_port $url $destdir]
1118        }
1119        https -
1120        http -
1121        ftp {
1122            return [macports::fetch_port $url]
1123        }
1124        default {
1125            return -code error "Unsupported protocol $protocol"
1126        }
1127    }
1128}
1129
1130##
1131# Get the path to the _resources directory of the source
1132#
1133# If the file is not available in the current source, it will fall back to the
1134# default source. This behavior is controlled by the fallback parameter.
1135#
1136# @param url port url
1137# @param path path in _resources we are interested in
1138# @param fallback fall back to the default source tree
1139# @return path to the _resources directory or the path to the fallback
1140proc macports::getportresourcepath {url {path ""} {fallback yes}} {
1141    global macports::sources_default
1142
1143    set protocol [getprotocol $url]
1144
1145    switch -- ${protocol} {
1146        file {
1147            set proposedpath [file normalize [file join [getportdir $url] .. ..]]
1148        }
1149        default {
1150            set proposedpath [getsourcepath $url]
1151        }
1152    }
1153
1154    # append requested path
1155    set proposedpath [file join $proposedpath _resources $path]
1156
1157    if {$fallback == "yes" && ![file exists $proposedpath]} {
1158        return [getdefaultportresourcepath $path]
1159    }
1160
1161    return $proposedpath
1162}
1163
1164##
1165# Get the path to the _resources directory of the default source
1166#
1167# @param path path in _resources we are interested in
1168# @return path to the _resources directory of the default source
1169proc macports::getdefaultportresourcepath {{path ""}} {
1170    global macports::sources_default
1171
1172    set default_source_url [lindex ${sources_default} 0]
1173    if {[getprotocol $default_source_url] == "file"} {
1174        set proposedpath [getportdir $default_source_url]
1175    } else {
1176        set proposedpath [getsourcepath $default_source_url]
1177    }
1178
1179    # append requested path
1180    set proposedpath [file join $proposedpath _resources $path]
1181
1182    return $proposedpath
1183}
1184
1185
1186# mport_filtervariants
1187# returns the given list of variants with implicitly-set ones removed
1188proc mport_filtervariants {variations {warn yes}} {
1189    # Iterate through the variants, filtering out
1190    # implicit ones. At the moment, the only implicit variants are
1191    # platform variants.
1192    set filteredvariations {}
1193
1194    foreach {variation value} $variations {
1195        switch -regexp $variation {
1196            ^(pure)?darwin         -
1197            ^(free|net|open){1}bsd -
1198            ^i386                  -
1199            ^linux                 -
1200            ^macosx                -
1201            ^powerpc               -
1202            ^solaris               -
1203            ^sunos {
1204                if {$warn} {
1205                    ui_warn "Implicit variants should not be explicitly set or unset. $variation will be ignored."
1206                }
1207            }
1208            default {
1209                lappend filteredvariations $variation $value
1210            }
1211        }
1212    }
1213    return $filteredvariations
1214}
1215
1216
1217# mportopen
1218# Opens a MacPorts portfile specified by a URL.  The Portfile is
1219# opened with the given list of options and variations.  The result
1220# of this function should be treated as an opaque handle to a
1221# MacPorts Portfile.
1222
1223proc mportopen {porturl {options ""} {variations ""} {nocache ""}} {
1224    global macports::portdbpath macports::portconf macports::open_mports auto_path
1225
1226    # Look for an already-open MPort with the same URL.
1227    # XXX: should compare options and variations here too.
1228    # if found, return the existing reference and bump the refcount.
1229    if {$nocache != ""} {
1230        set mport {}
1231    } else {
1232        set mport [dlist_search $macports::open_mports porturl $porturl]
1233    }
1234    if {$mport != {}} {
1235        set refcnt [ditem_key $mport refcnt]
1236        incr refcnt
1237        ditem_key $mport refcnt $refcnt
1238        return $mport
1239    }
1240
1241    array set options_array $options
1242    if {[info exists options_array(portdir)]} {
1243        set portdir $options_array(portdir)
1244    } else {
1245        set portdir ""
1246    }
1247    if {![info exists options_array(ports_requested)]} {
1248        lappend options ports_requested 0
1249    }
1250
1251    set portpath [macports::getportdir $porturl $portdir]
1252    ui_debug "Changing to port directory: $portpath"
1253    cd $portpath
1254    if {![file isfile Portfile]} {
1255        return -code error "Could not find Portfile in $portpath"
1256    }
1257
1258    set workername [interp create]
1259
1260    set mport [ditem_create]
1261    lappend macports::open_mports $mport
1262    ditem_key $mport porturl $porturl
1263    ditem_key $mport portpath $portpath
1264    ditem_key $mport workername $workername
1265    ditem_key $mport options $options
1266    ditem_key $mport variations $variations
1267    ditem_key $mport refcnt 1
1268
1269    macports::worker_init $workername $portpath $porturl [macports::getportbuildpath $portpath] $options $variations
1270
1271    $workername eval source Portfile
1272
1273    # add the default universal variant if appropriate, and set up flags that
1274    # are conditional on whether universal is set
1275    $workername eval universal_setup
1276
1277    # evaluate the variants
1278    if {[$workername eval eval_variants variations] != 0} {
1279        mportclose $mport
1280        error "Error evaluating variants"
1281    }
1282
1283    ditem_key $mport provides [$workername eval return \$name]
1284
1285    return $mport
1286}
1287
1288# mportopen_installed
1289# opens a portfile stored in the registry
1290proc mportopen_installed {name version revision variants options} {
1291    global macports::registry.format macports::registry.path
1292    if {${registry.format} != "receipt_sqlite"} {
1293        return -code error "mportopen_installed requires sqlite registry"
1294    }
1295    set regref [lindex [registry::entry imaged $name $version $revision $variants] 0]
1296    set portfile_dir [file join ${registry.path} registry portfiles $name "${version}_${revision}${variants}"]
1297    file mkdir $portfile_dir
1298    set fd [open "${portfile_dir}/Portfile" w]
1299    puts $fd [$regref portfile]
1300    close $fd
1301    file mtime "${portfile_dir}/Portfile" [$regref date]
1302
1303    set variations {}
1304    set minusvariant [lrange [split [$regref negated_variants] -] 1 end]
1305    set plusvariant [lrange [split [$regref variants] +] 1 end]
1306    foreach v $plusvariant {
1307        lappend variations $v "+"
1308    }
1309    foreach v $minusvariant {
1310        lappend variations $v "-"
1311    }
1312    set variations [mport_filtervariants $variations no]
1313   
1314    return [mportopen "file://${portfile_dir}/" $options $variations]
1315}
1316
1317# mportclose_installed
1318# close mport opened with mportopen_installed and clean up associated files
1319proc mportclose_installed {mport} {
1320    global macports::registry.path
1321    foreach key {name version revision portvariants} {
1322        set $key [_mportkey $mport $key]
1323    }
1324    mportclose $mport
1325    set portfiles_dir [file join ${registry.path} registry portfiles $name]
1326    set portfile [file join $portfiles_dir "${version}_${revision}${portvariants}" Portfile]
1327    file delete -force $portfile [file dirname $portfile]
1328    if {[llength [glob -nocomplain -directory $portfiles_dir *]] == 0} {
1329        file delete -force $portfiles_dir
1330    }
1331}
1332
1333# Traverse a directory with ports, calling a function on the path of ports
1334# (at the second depth).
1335# I.e. the structure of dir shall be:
1336# category/port/
1337# with a Portfile file in category/port/
1338#
1339# func:     function to call on every port directory (it is passed
1340#           category/port/ as its parameter)
1341# root:     the directory with all the categories directories.
1342proc mporttraverse {func {root .}} {
1343    # Save the current directory
1344    set pwd [pwd]
1345
1346    # Join the root.
1347    set pathToRoot [file join $pwd $root]
1348
1349    # Go to root because some callers expects us to be there.
1350    cd $pathToRoot
1351
1352    foreach category [lsort -increasing -unique [readdir $root]] {
1353        set pathToCategory [file join $root $category]
1354        # process the category dirs but not _resources
1355        if {[file isdirectory $pathToCategory] && [string index [file tail $pathToCategory] 0] != "_"} {
1356            # Iterate on port directories.
1357            foreach port [lsort -increasing -unique [readdir $pathToCategory]] {
1358                set pathToPort [file join $pathToCategory $port]
1359                if {[file isdirectory $pathToPort] &&
1360                  [file exists [file join $pathToPort "Portfile"]]} {
1361                    # Call the function.
1362                    $func [file join $category $port]
1363
1364                    # Restore the current directory because some
1365                    # functions changes it.
1366                    cd $pathToRoot
1367                }
1368            }
1369        }
1370    }
1371
1372    # Restore the current directory.
1373    cd $pwd
1374}
1375
1376### _mportsearchpath is private; subject to change without notice
1377
1378# depregex -> regex on the filename to find.
1379# search_path -> directories to search
1380# executable -> whether we want to check that the file is executable by current
1381#               user or not.
1382proc _mportsearchpath {depregex search_path {executable 0} {return_match 0}} {
1383    set found 0
1384    foreach path $search_path {
1385        if {![file isdirectory $path]} {
1386            continue
1387        }
1388
1389        if {[catch {set filelist [readdir $path]} result]} {
1390            return -code error "$result ($path)"
1391        }
1392
1393        foreach filename $filelist {
1394            if {[regexp $depregex $filename] &&
1395              (($executable == 0) || [file executable [file join $path $filename]])} {
1396                ui_debug "Found Dependency: path: $path filename: $filename regex: $depregex"
1397                set found 1
1398                break
1399            }
1400        }
1401    }
1402    if {$return_match} {
1403        if {$found} {
1404            return [file join $path $filename]
1405        } else {
1406            return ""
1407        }
1408    } else {
1409        return $found
1410    }
1411}
1412
1413
1414### _mportinstalled is private; may change without notice
1415
1416# Determine if a port is already *installed*, as in "in the registry".
1417proc _mportinstalled {mport} {
1418    # Check for the presence of the port in the registry
1419    set workername [ditem_key $mport workername]
1420    return [$workername eval registry_exists_for_name \${name}]
1421}
1422
1423# Determine if a port is active (only for image mode)
1424proc _mportactive {mport} {
1425    set workername [ditem_key $mport workername]
1426    if {[catch {set reslist [$workername eval registry_active \${name}]}]} {
1427        return 0
1428    } else {
1429        return [expr [llength $reslist] > 0]
1430    }
1431}
1432
1433# Determine if the named port is active (only for image mode)
1434proc _portnameactive {portname} {
1435    if {[catch {set reslist [registry::active $portname]}]} {
1436        return 0
1437    } else {
1438        return [expr [llength $reslist] > 0]
1439    }
1440}
1441
1442### _mportispresent is private; may change without notice
1443
1444# Determine if some depspec is satisfied or if the given port is installed
1445# (and active, if we're in image mode).
1446# We actually start with the registry (faster?)
1447#
1448# mport     the port declaring the dep (context in which to evaluate $prefix etc)
1449# depspec   the dependency test specification (path, bin, lib, etc.)
1450proc _mportispresent {mport depspec} {
1451    set portname [lindex [split $depspec :] end]
1452    ui_debug "Searching for dependency: $portname"
1453    if {[string equal ${macports::registry.installtype} "image"]} {
1454        set res [_portnameactive $portname]
1455    } else {
1456        set res [registry::entry_exists_for_name $portname]
1457    }
1458    if {$res != 0} {
1459        ui_debug "Found Dependency: receipt exists for $portname"
1460        return 1
1461    } else {
1462        # The receipt test failed, use one of the depspec regex mechanisms
1463        ui_debug "Didn't find receipt, going to depspec regex for: $portname"
1464        set workername [ditem_key $mport workername]
1465        set type [lindex [split $depspec :] 0]
1466        switch $type {
1467            lib { return [$workername eval _libtest $depspec] }
1468            bin { return [$workername eval _bintest $depspec] }
1469            path { return [$workername eval _pathtest $depspec] }
1470            port { return 0 }
1471            default {return -code error "unknown depspec type: $type"}
1472        }
1473        return 0
1474    }
1475}
1476
1477### _mportconflictsinstalled is private; may change without notice
1478
1479# Determine if the port, per the conflicts option, has any conflicts with
1480# what is installed.
1481#
1482# mport   the port to check for conflicts
1483# Returns a list of which installed ports conflict, or an empty list if none
1484proc _mportconflictsinstalled {mport conflictinfo} {
1485    set conflictlist {}
1486    if {[llength $conflictinfo] > 0} {
1487        ui_debug "Checking for conflicts against [_mportkey $mport name]"
1488        foreach conflictport ${conflictinfo} {
1489            if {[_mportispresent $mport port:${conflictport}]} {
1490                lappend conflictlist $conflictport
1491            }
1492        }
1493    } else {
1494        ui_debug "[_mportkey $mport name] has no conflicts"
1495    }
1496
1497    return $conflictlist
1498}
1499
1500
1501### _mportexec is private; may change without notice
1502
1503proc _mportexec {target mport} {
1504    set portname [_mportkey $mport name]
1505    macports::push_log $mport
1506    # xxx: set the work path?
1507    set workername [ditem_key $mport workername]
1508    if {![catch {$workername eval check_variants $target} result] && $result == 0 &&
1509        ![catch {$workername eval eval_targets $target} result] && $result == 0} {
1510        # If auto-clean mode, clean-up after dependency install
1511        if {[string equal ${macports::portautoclean} "yes"]} {
1512            # Make sure we are back in the port path before clean
1513            # otherwise if the current directory had been changed to
1514            # inside the port,  the next port may fail when trying to
1515            # install because [pwd] will return a "no file or directory"
1516            # error since the directory it was in is now gone.
1517            set portpath [ditem_key $mport portpath]
1518            catch {cd $portpath}
1519            $workername eval eval_targets clean
1520        }
1521        macports::pop_log
1522        return 0
1523    } else {
1524        # An error occurred.
1525        global ::debuglogname
1526        if {[info exists ::debuglogname]} {
1527            ui_msg "Log for $portname is at: $::debuglogname"
1528        }
1529        macports::pop_log
1530        return 1
1531    }
1532}
1533
1534# mportexec
1535# Execute the specified target of the given mport.
1536proc mportexec {mport target} {
1537    global macports::registry.installtype
1538
1539    set workername [ditem_key $mport workername]
1540
1541    # check variants
1542    if {[$workername eval check_variants $target] != 0} {
1543        return 1
1544    }
1545    set portname [_mportkey $mport name]
1546    if {$target != "clean"} {
1547        macports::push_log $mport
1548    }
1549
1550    # Before we build the port, we must build its dependencies.
1551    # XXX: need a more general way of comparing against targets
1552    set dlist {}
1553    if {   $target == "fetch" || $target == "checksum"
1554        || $target == "extract" || $target == "patch"
1555        || $target == "configure" || $target == "build"
1556        || $target == "test"
1557        || $target == "destroot" || $target == "install"
1558        || $target == "archive"
1559        || $target == "dmg" || $target == "mdmg"
1560        || $target == "pkg" || $target == "mpkg"
1561        || $target == "rpm" || $target == "dpkg"
1562        || $target == "srpm"|| $target == "portpkg" } {
1563
1564        # upgrade dependencies that are already installed
1565        if {![macports::global_option_isset ports_nodeps]} {
1566            macports::_upgrade_mport_deps $mport $target
1567        }
1568
1569        ui_msg -nonewline "--->  Computing dependencies for [_mportkey $mport name]"
1570        if {[macports::ui_isset ports_debug]} {
1571            # play nice with debug messages
1572            ui_msg ""
1573        }
1574        if {[mportdepends $mport $target] != 0} {
1575            return 1
1576        }
1577        if {![macports::ui_isset ports_debug]} {
1578            ui_msg ""
1579        }
1580
1581        # Select out the dependents along the critical path,
1582        # but exclude this mport, we might not be installing it.
1583        set dlist [dlist_append_dependents $macports::open_mports $mport {}]
1584
1585        dlist_delete dlist $mport
1586
1587        # install them
1588        # xxx: as with below, this is ugly.  and deps need to be fixed to
1589        # understand Port Images before this can get prettier
1590        if { [string equal ${macports::registry.installtype} "image"] } {
1591            set result [dlist_eval $dlist _mportactive [list _mportexec "activate"]]
1592        } else {
1593            set result [dlist_eval $dlist _mportinstalled [list _mportexec "install"]]
1594        }
1595
1596        if {$result != {}} {
1597            set errstring "The following dependencies failed to build:"
1598            foreach ditem $result {
1599                append errstring " [ditem_key $ditem provides]"
1600            }
1601            ui_error $errstring
1602            return 1
1603        }
1604
1605        # Close the dependencies, we're done installing them.
1606        foreach ditem $dlist {
1607            mportclose $ditem
1608        }
1609    }
1610
1611    set clean 0
1612    if {[string equal $target "install"]} {
1613        # mark port as explicitly requested
1614        $workername eval set user_options(ports_requested) 1
1615
1616        # If we're doing image installs, then we should activate after install
1617        # xxx: This isn't pretty
1618        if { [string equal ${macports::registry.installtype} "image"] } {
1619            set target activate
1620        }
1621    }
1622    if {[string equal ${macports::portautoclean} "yes"] && ([string equal $target "install"] || [string equal $target "activate"])} {
1623        # If we're doing an install, check if we should clean after
1624        set clean 1
1625    }
1626
1627    # Build this port with the specified target
1628    set result [$workername eval eval_targets $target]
1629
1630    # If auto-clean mode and successful install, clean-up after install
1631    if {$result == 0 && $clean == 1} {
1632        # Make sure we are back in the port path, just in case
1633        set portpath [ditem_key $mport portpath]
1634        catch {cd $portpath}
1635        $workername eval eval_targets clean
1636    }
1637   
1638    global ::debuglogname
1639    if {[info exists ::debuglogname]} {
1640        if {$result != 0 && ![macports::ui_isset ports_quiet]} {
1641            ui_msg "Log for $portname is at: $::debuglogname"
1642        }
1643        macports::pop_log
1644    }
1645
1646    return $result
1647}
1648
1649# upgrade any dependencies of mport that are installed and needed for target
1650proc macports::_upgrade_mport_deps {mport target} {
1651    set options [ditem_key $mport options]
1652    set deptypes [macports::_deptypes_for_target $target]
1653    array set portinfo [mportinfo $mport]
1654    set depends {}
1655    array set depscache {}
1656
1657    foreach deptype $deptypes {
1658        # Add to the list of dependencies if the option exists and isn't empty.
1659        if {[info exists portinfo($deptype)] && $portinfo($deptype) != ""} {
1660            set depends [concat $depends $portinfo($deptype)]
1661        }
1662    }
1663   
1664    foreach depspec $depends {
1665        set workername [ditem_key $mport workername]
1666        set dep_portname [$workername eval _get_dep_port $depspec]
1667        if {$dep_portname != "" && ![info exists depscache(port:$dep_portname)] && [registry::entry_exists_for_name $dep_portname]} {
1668            set status [macports::upgrade $dep_portname "port:$dep_portname" {} $options depscache]
1669            # status 2 means the port was not found in the index
1670            if {$status != 0 && $status != 2 && ![macports::ui_isset ports_processall]} {
1671                return -code error "upgrade $dep_portname failed"
1672            }
1673        }
1674    }
1675}
1676
1677proc macports::getsourcepath {url} {
1678    global macports::portdbpath
1679
1680    set source_path [split $url ://]
1681
1682    if {[_source_is_snapshot $url]} {
1683        # daily snapshot tarball
1684        return [file join $portdbpath sources [join [lrange $source_path 3 end-1] /] ports]
1685    }
1686
1687    return [file join $portdbpath sources [lindex $source_path 3] [lindex $source_path 4] [lindex $source_path 5]]
1688}
1689
1690##
1691# Checks whether a supplied source URL is for a daily snapshot tarball
1692# (private)
1693#
1694# @param url source URL to check
1695# @return a list containing filename and extension or an empty list
1696proc _source_is_snapshot {url {filename ""} {extension ""}} {
1697    upvar $filename myfilename
1698    upvar $extension myextension
1699
1700    if {[regexp {^(?:https?|ftp)://.+/(.+\.(tar\.gz|tar\.bz2))$} $url -> f e]} {
1701        set myfilename $f
1702        set myextension $e
1703
1704        return 1
1705    }
1706
1707    return 0
1708}
1709
1710proc macports::getportbuildpath {id} {
1711    global macports::portdbpath
1712    regsub {://} $id {.} port_path
1713    regsub -all {/} $port_path {_} port_path
1714    return [file join $portdbpath build $port_path]
1715}
1716
1717proc macports::getportlogpath {id} {
1718    global macports::portdbpath
1719    regsub {://} $id {.} port_path
1720    regsub -all {/} $port_path {_} port_path
1721    return [file join $portdbpath logs $port_path]
1722}
1723
1724proc macports::getportworkpath_from_buildpath {portbuildpath} {
1725    return [file join $portbuildpath work]
1726}
1727
1728proc macports::getportworkpath_from_portdir {portpath} {
1729    return [macports::getportworkpath_from_buildpath [macports::getportbuildpath $portpath]]
1730}
1731
1732proc macports::getindex {source} {
1733    # Special case file:// sources
1734    if {[macports::getprotocol $source] == "file"} {
1735        return [file join [macports::getportdir $source] PortIndex]
1736    }
1737
1738    return [file join [macports::getsourcepath $source] PortIndex]
1739}
1740
1741proc mportsync {{optionslist {}}} {
1742    global macports::sources macports::portdbpath macports::rsync_options tcl_platform
1743    global macports::portverbose
1744    global macports::autoconf::rsync_path
1745    array set options $optionslist
1746
1747    set numfailed 0
1748
1749    ui_debug "Synchronizing ports tree(s)"
1750    foreach source $sources {
1751        set flags [lrange $source 1 end]
1752        set source [lindex $source 0]
1753        if {[lsearch -exact $flags nosync] != -1} {
1754            ui_debug "Skipping $source"
1755            continue
1756        }
1757        ui_info "Synchronizing local ports tree from $source"
1758        switch -regexp -- [macports::getprotocol $source] {
1759            {^file$} {
1760                set portdir [macports::getportdir $source]
1761                if {[file exists $portdir/.svn]} {
1762                    set svn_commandline "[macports::findBinary svn] update --non-interactive ${portdir}"
1763                    ui_debug $svn_commandline
1764                    if {
1765                        [catch {
1766                            if {[getuid] == 0} {
1767                                set euid [geteuid]
1768                                set egid [getegid]
1769                                ui_debug "changing euid/egid - current euid: $euid - current egid: $egid"
1770                                setegid [name_to_gid [file attributes $portdir -group]]
1771                                seteuid [name_to_uid [file attributes $portdir -owner]]
1772                            }
1773                            system $svn_commandline
1774                            if {[getuid] == 0} {
1775                                seteuid $euid
1776                                setegid $egid
1777                            }
1778                        }]
1779                    } {
1780                        ui_debug "$::errorInfo"
1781                        ui_error "Synchronization of the local ports tree failed doing an svn update"
1782                        incr numfailed
1783                        continue
1784                    }
1785                }
1786            }
1787            {^mports$} {
1788                macports::index::sync $macports::portdbpath $source
1789            }
1790            {^rsync$} {
1791                # Where to, boss?
1792                set destdir [file dirname [macports::getindex $source]]
1793                file mkdir $destdir
1794                # Keep rsync happy with a trailing slash
1795                if {[string index $source end] != "/"} {
1796                    set source "${source}/"
1797                }
1798                # Do rsync fetch
1799                set rsync_commandline "${macports::autoconf::rsync_path} ${rsync_options} ${source} ${destdir}"
1800                ui_debug $rsync_commandline
1801                if {[catch {system $rsync_commandline}]} {
1802                    ui_error "Synchronization of the local ports tree failed doing rsync"
1803                    incr numfailed
1804                    continue
1805                }
1806                if {[catch {system "chmod -R a+r \"$destdir\""}]} {
1807                    ui_warn "Setting world read permissions on parts of the ports tree failed, need root?"
1808                }
1809            }
1810            {^https?$|^ftp$} {
1811                if {[_source_is_snapshot $source filename extension]} {
1812                    # sync a daily port snapshot tarball
1813                    set indexfile [macports::getindex $source]
1814                    set destdir [file dirname $indexfile]
1815                    set tarpath [file join [file normalize [file join $destdir ..]] $filename]
1816
1817                    set updated 1
1818                    if {[file isdirectory $destdir]} {
1819                        set moddate [file mtime $destdir]
1820                        if {[catch {set updated [curl isnewer $source $moddate]} error]} {
1821                            ui_warn "Cannot check if $source was updated, ($error)"
1822                        }
1823                    }
1824
1825                    if {(![info exists options(ports_force)] || $options(ports_force) != "yes") && $updated <= 0} {
1826                        ui_info "No updates for $source"
1827                        continue
1828                    }
1829
1830                    file mkdir [file dirname $indexfile]
1831
1832                    set verboseflag {}
1833                    if {$macports::portverbose == "yes"} {
1834                        set verboseflag "-v"
1835                    }
1836
1837                    if {[catch {eval curl fetch $verboseflag {$source} {$tarpath}} error]} {
1838                        ui_error "Fetching $source failed ($error)"
1839                        incr numfailed
1840                        continue
1841                    }
1842
1843                    set extflag {}
1844                    switch $extension {
1845                        {tar.gz} {
1846                            set extflag "-z"
1847                        }
1848                        {tar.bz2} {
1849                            set extflag "-j"
1850                        }
1851                    }
1852
1853                    set tar [macports::findBinary tar $macports::autoconf::tar_path]
1854                    if { [catch { system "cd $destdir/.. && $tar ${verboseflag} ${extflag} -xf $filename" } error] } {
1855                        ui_error "Extracting $source failed ($error)"
1856                        incr numfailed
1857                        continue
1858                    }
1859
1860                    if {[catch {system "chmod -R a+r \"$destdir\""}]} {
1861                        ui_warn "Setting world read permissions on parts of the ports tree failed, need root?"
1862                    }
1863
1864                    file delete $tarpath
1865                } else {
1866                    # sync just a PortIndex file
1867                    set indexfile [macports::getindex $source]
1868                    file mkdir [file dirname $indexfile]
1869                    curl fetch ${source}/PortIndex $indexfile
1870                    curl fetch ${source}/PortIndex.quick ${indexfile}.quick
1871                }
1872            }
1873            default {
1874                ui_warn "Unknown synchronization protocol for $source"
1875            }
1876        }
1877    }
1878
1879    if {$numfailed > 0} {
1880        return -code error "Synchronization of $numfailed source(s) failed"
1881    }
1882}
1883
1884proc mportsearch {pattern {case_sensitive yes} {matchstyle regexp} {field name}} {
1885    global macports::portdbpath macports::sources
1886    set matches [list]
1887    set easy [expr { $field == "name" }]
1888
1889    set found 0
1890    foreach source $sources {
1891        set source [lindex $source 0]
1892        set protocol [macports::getprotocol $source]
1893        if {$protocol == "mports"} {
1894            set res [macports::index::search $macports::portdbpath $source [list name $pattern]]
1895            eval lappend matches $res
1896        } else {
1897            if {[catch {set fd [open [macports::getindex $source] r]} result]} {
1898                ui_warn "Can't open index file for source: $source"
1899            } else {
1900                try {
1901                    incr found 1
1902                    while {[gets $fd line] >= 0} {
1903                        array unset portinfo
1904                        set name [lindex $line 0]
1905                        set len [lindex $line 1]
1906                        set line [read $fd $len]
1907
1908                        if {$easy} {
1909                            set target $name
1910                        } else {
1911                            array set portinfo $line
1912                            if {![info exists portinfo($field)]} continue
1913                            set target $portinfo($field)
1914                        }
1915
1916                        switch $matchstyle {
1917                            exact {
1918                                set matchres [expr 0 == ( {$case_sensitive == "yes"} ? [string compare $pattern $target] : [string compare -nocase $pattern $target] )]
1919                            }
1920                            glob {
1921                                set matchres [expr {$case_sensitive == "yes"} ? [string match $pattern $target] : [string match -nocase $pattern $target]]
1922                            }
1923                            regexp -
1924                            default {
1925                                set matchres [expr {$case_sensitive == "yes"} ? [regexp -- $pattern $target] : [regexp -nocase -- $pattern $target]]
1926                            }
1927                        }
1928
1929                        if {$matchres == 1} {
1930                            if {$easy} {
1931                                array set portinfo $line
1932                            }
1933                            switch $protocol {
1934                                rsync {
1935                                    # Rsync files are local
1936                                    set source_url "file://[macports::getsourcepath $source]"
1937                                }
1938                                https -
1939                                http -
1940                                ftp {
1941                                    if {[_source_is_snapshot $source filename extension]} {
1942                                        # daily snapshot tarball
1943                                        set source_url "file://[macports::getsourcepath $source]"
1944                                    } else {
1945                                        # default action
1946                                        set source_url $source
1947                                    }
1948                                }
1949                                default {
1950                                    set source_url $source
1951                                }
1952                            }
1953                            if {[info exists portinfo(portarchive)]} {
1954                                set porturl ${source_url}/$portinfo(portarchive)
1955                            } elseif {[info exists portinfo(portdir)]} {
1956                                set porturl ${source_url}/$portinfo(portdir)
1957                            }
1958                            if {[info exists porturl]} {
1959                                lappend line porturl $porturl
1960                                ui_debug "Found port in $porturl"
1961                            } else {
1962                                ui_debug "Found port info: $line"
1963                            }
1964                            lappend matches $name
1965                            lappend matches $line
1966                        }
1967                    }
1968                } catch {*} {
1969                    ui_warn "It looks like your PortIndex file for $source may be corrupt."
1970                    throw
1971                } finally {
1972                    close $fd
1973                }
1974            }
1975        }
1976    }
1977    if {!$found} {
1978        return -code error "No index(es) found! Have you synced your source indexes?"
1979    }
1980
1981    return $matches
1982}
1983
1984# Returns the PortInfo for a single named port. The info comes from the
1985# PortIndex, and name matching is case-insensitive. Unlike mportsearch, only
1986# the first match is returned, but the return format is otherwise identical.
1987# The advantage is that mportlookup is much faster than mportsearch, due to
1988# the use of the quick index.
1989proc mportlookup {name} {
1990    global macports::portdbpath macports::sources
1991
1992    set sourceno 0
1993    set matches [list]
1994    foreach source $sources {
1995        set source [lindex $source 0]
1996        set protocol [macports::getprotocol $source]
1997        if {$protocol != "mports"} {
1998            global macports::quick_index
1999            if {![info exists quick_index($sourceno,[string tolower $name])]} {
2000                incr sourceno 1
2001                continue
2002            }
2003            # The quick index is keyed on the port name, and provides the
2004            # offset in the main PortIndex where the given port's PortInfo
2005            # line can be found.
2006            set offset $quick_index($sourceno,[string tolower $name])
2007            incr sourceno 1
2008            if {[catch {set fd [open [macports::getindex $source] r]} result]} {
2009                ui_warn "Can't open index file for source: $source"
2010            } else {
2011                try {
2012                    seek $fd $offset
2013                    gets $fd line
2014                    set name [lindex $line 0]
2015                    set len [lindex $line 1]
2016                    set line [read $fd $len]
2017
2018                    array set portinfo $line
2019
2020                    switch $protocol {
2021                        rsync {
2022                            set source_url "file://[macports::getsourcepath $source]"
2023                        }
2024                        https -
2025                        http -
2026                        ftp {
2027                            if {[_source_is_snapshot $source filename extension]} {
2028                                set source_url "file://[macports::getsourcepath $source]"
2029                             } else {
2030                                set source_url $source
2031                             }
2032                        }
2033                        default {
2034                            set source_url $source
2035                        }
2036                    }
2037                    if {[info exists portinfo(portarchive)]} {
2038                        set porturl ${source_url}/$portinfo(portarchive)
2039                    } elseif {[info exists portinfo(portdir)]} {
2040                        set porturl ${source_url}/$portinfo(portdir)
2041                    }
2042                    if {[info exists porturl]} {
2043                        lappend line porturl $porturl
2044                    }
2045                    lappend matches $name
2046                    lappend matches $line
2047                    close $fd
2048                    set fd -1
2049                } catch {*} {
2050                    ui_warn "It looks like your PortIndex file for $source may be corrupt."
2051                } finally {
2052                    if {$fd != -1} {
2053                        close $fd
2054                    }
2055                }
2056                if {[llength $matches] > 0} {
2057                    break
2058                }
2059            }
2060        } else {
2061            set res [macports::index::search $macports::portdbpath $source [list name $name]]
2062            if {[llength $res] > 0} {
2063                eval lappend matches $res
2064                break
2065            }
2066        }
2067    }
2068
2069    return $matches
2070}
2071
2072# Returns all ports in the indices. Faster than 'mportsearch .*'
2073proc mportlistall {args} {
2074    global macports::portdbpath macports::sources
2075    set matches [list]
2076
2077    set found 0
2078    foreach source $sources {
2079        set source [lindex $source 0]
2080        set protocol [macports::getprotocol $source]
2081        if {$protocol != "mports"} {
2082            if {![catch {set fd [open [macports::getindex $source] r]} result]} {
2083                try {
2084                    incr found 1
2085                    while {[gets $fd line] >= 0} {
2086                        array unset portinfo
2087                        set name [lindex $line 0]
2088                        set len [lindex $line 1]
2089                        set line [read $fd $len]
2090
2091                        array set portinfo $line
2092
2093                        switch $protocol {
2094                            rsync {
2095                                set source_url "file://[macports::getsourcepath $source]"
2096                            }
2097                            https -
2098                            http -
2099                            ftp {
2100                                if {[_source_is_snapshot $source filename extension]} {
2101                                    set source_url "file://[macports::getsourcepath $source]"
2102                                } else {
2103                                    set source_url $source
2104                                }
2105                            }
2106                            default {
2107                                set source_url $source
2108                            }
2109                        }
2110                        if {[info exists portinfo(portdir)]} {
2111                            set porturl ${source_url}/$portinfo(portdir)
2112                        } elseif {[info exists portinfo(portarchive)]} {
2113                            set porturl ${source_url}/$portinfo(portarchive)
2114                        }
2115                        if {[info exists porturl]} {
2116                            lappend line porturl $porturl
2117                        }
2118                        lappend matches $name $line
2119                    }
2120                } catch {*} {
2121                    ui_warn "It looks like your PortIndex file for $source may be corrupt."
2122                    throw
2123                } finally {
2124                    close $fd
2125                }
2126            } else {
2127                ui_warn "Can't open index file for source: $source"
2128            }
2129        } else {
2130            set res [macports::index::search $macports::portdbpath $source [list name .*]]
2131            eval lappend matches $res
2132        }
2133    }
2134    if {!$found} {
2135        return -code error "No index(es) found! Have you synced your source indexes?"
2136    }
2137
2138    return $matches
2139}
2140
2141
2142# Loads PortIndex.quick from each source into the quick_index, generating
2143# it first if necessary.
2144proc _mports_load_quickindex {args} {
2145    global macports::sources macports::quick_index
2146
2147    set sourceno 0
2148    foreach source $sources {
2149        unset -nocomplain quicklist
2150        # chop off any tags
2151        set source [lindex $source 0]
2152        set index [macports::getindex $source]
2153        if {![file exists ${index}]} {
2154            continue
2155        }
2156        if {![file exists ${index}.quick]} {
2157            ui_warn "No quick index file found, attempting to generate one for source: $source"
2158            if {[catch {set quicklist [mports_generate_quickindex ${index}]}]} {
2159                continue
2160            }
2161        }
2162        # only need to read the quick index file if we didn't just update it
2163        if {![info exists quicklist]} {
2164            if {[catch {set fd [open ${index}.quick r]} result]} {
2165                ui_warn "Can't open quick index file for source: $source"
2166                continue
2167            } else {
2168                set quicklist [read $fd]
2169                close $fd
2170            }
2171        }
2172        foreach entry [split $quicklist "\n"] {
2173            set quick_index($sourceno,[lindex $entry 0]) [lindex $entry 1]
2174        }
2175        incr sourceno 1
2176    }
2177    if {!$sourceno} {
2178        ui_warn "No index(es) found! Have you synced your source indexes?"
2179    }
2180}
2181
2182proc mports_generate_quickindex {index} {
2183    if {[catch {set indexfd [open ${index} r]} result] || [catch {set quickfd [open ${index}.quick w]} result]} {
2184        ui_warn "Can't open index file: $index"
2185        return -code error
2186    } else {
2187        try {
2188            set offset [tell $indexfd]
2189            set quicklist ""
2190            while {[gets $indexfd line] >= 0} {
2191                if {[llength $line] != 2} {
2192                    continue
2193                }
2194                set name [lindex $line 0]
2195                append quicklist "[string tolower $name] ${offset}\n"
2196
2197                set len [lindex $line 1]
2198                read $indexfd $len
2199                set offset [tell $indexfd]
2200            }
2201            puts -nonewline $quickfd $quicklist
2202        } catch {*} {
2203            ui_warn "It looks like your PortIndex file $index may be corrupt."
2204            throw
2205        } finally {
2206            close $indexfd
2207            close $quickfd
2208        }
2209    }
2210    if {[info exists quicklist]} {
2211        return $quicklist
2212    } else {
2213        ui_warn "Failed to generate quick index for: $index"
2214        return -code error
2215    }
2216}
2217
2218proc mportinfo {mport} {
2219    set workername [ditem_key $mport workername]
2220    return [$workername eval array get PortInfo]
2221}
2222
2223proc mportclose {mport} {
2224    global macports::open_mports
2225    set refcnt [ditem_key $mport refcnt]
2226    incr refcnt -1
2227    ditem_key $mport refcnt $refcnt
2228    if {$refcnt == 0} {
2229        dlist_delete macports::open_mports $mport
2230        set workername [ditem_key $mport workername]
2231        interp delete $workername
2232        ditem_delete $mport
2233    }
2234}
2235
2236##### Private Depspec API #####
2237# This API should be considered work in progress and subject to change without notice.
2238##### "
2239
2240# _mportkey
2241# - returns a variable from the port's interpreter
2242
2243proc _mportkey {mport key} {
2244    set workername [ditem_key $mport workername]
2245    return [$workername eval "return \$${key}"]
2246}
2247
2248# mportdepends builds the list of mports which the given port depends on.
2249# This list is added to $mport.
2250# This list actually depends on the target.
2251# This method can optionally recurse through the dependencies, looking for
2252#   dependencies of dependencies.
2253# This method can optionally cut the search when ports are already installed or
2254#   the dependencies are satisfied.
2255#
2256# mport -> mport item
2257# target -> target to consider the dependency for
2258# recurseDeps -> if the search should be recursive
2259# skipSatisfied -> cut the search tree when encountering installed/satisfied
2260#                  dependencies ports.
2261# accDeps -> accumulator for recursive calls
2262# return 0 if everything was ok, an non zero integer otherwise.
2263proc mportdepends {mport {target ""} {recurseDeps 1} {skipSatisfied 1}} {
2264
2265    array set portinfo [mportinfo $mport]
2266    set depends {}
2267    set deptypes {}
2268
2269    # progress indicator
2270    if {![macports::ui_isset ports_debug]} {
2271        ui_info -nonewline "."
2272        flush stdout
2273    }
2274   
2275    if {[info exists portinfo(conflicts)] && ($target == "" || $target == "install")} {
2276        set conflictports [_mportconflictsinstalled $mport $portinfo(conflicts)]
2277        if {[llength ${conflictports}] != 0} {
2278            if {[macports::global_option_isset ports_force]} {
2279                ui_warn "Force option set; installing $portinfo(name) despite conflicts with: ${conflictports}"
2280            } else {
2281                return -code error "Can't install $portinfo(name) because conflicting ports are installed: ${conflictports}"
2282            }
2283        }
2284    }
2285
2286    set deptypes [macports::_deptypes_for_target $target]
2287
2288    # Gather the dependencies for deptypes
2289    foreach deptype $deptypes {
2290        # Add to the list of dependencies if the option exists and isn't empty.
2291        if {[info exists portinfo($deptype)] && $portinfo($deptype) != ""} {
2292            set depends [concat $depends $portinfo($deptype)]
2293        }
2294    }
2295
2296    set subPorts {}
2297    set options [ditem_key $mport options]
2298    set variations [ditem_key $mport variations]
2299
2300    foreach depspec $depends {
2301        # Is that dependency satisfied or this port installed?
2302        # If we don't skip or if it is not, add it to the list.
2303        if {!$skipSatisfied || ![_mportispresent $mport $depspec]} {
2304            # grab the portname portion of the depspec
2305            set dep_portname [lindex [split $depspec :] end]
2306
2307            # Find the porturl
2308            if {[catch {set res [mportlookup $dep_portname]} error]} {
2309                global errorInfo
2310                ui_debug "$errorInfo"
2311                ui_error "Internal error: port lookup failed: $error"
2312                return 1
2313            }
2314
2315            array unset portinfo
2316            array set portinfo [lindex $res 1]
2317            if {![info exists portinfo(porturl)]} {
2318                if {![macports::ui_isset ports_debug]} {
2319                    ui_msg ""
2320                }
2321                ui_error "Dependency '$dep_portname' not found."
2322                return 1
2323            }
2324
2325            # Figure out the subport. Check the open_mports list first, since
2326            # we potentially leak mport references if we mportopen each time,
2327            # because mportexec only closes each open mport once.
2328            set subport [dlist_search $macports::open_mports porturl $portinfo(porturl)]
2329            if {$subport == {}} {
2330                # We haven't opened this one yet.
2331                set subport [mportopen $portinfo(porturl) $options $variations]
2332                if {$recurseDeps} {
2333                    # Add to the list we need to recurse on.
2334                    lappend subPorts $subport
2335                }
2336            }
2337
2338            # Append the sub-port's provides to the port's requirements list.
2339            ditem_append_unique $mport requires "[ditem_key $subport provides]"
2340        }
2341    }
2342
2343    # Loop on the subports.
2344    if {$recurseDeps} {
2345        foreach subport $subPorts {
2346            # Sub ports should be installed (all dependencies must be satisfied).
2347            set res [mportdepends $subport "" $recurseDeps $skipSatisfied]
2348            if {$res != 0} {
2349                return $res
2350            }
2351        }
2352    }
2353
2354    return 0
2355}
2356
2357# Determine dependency types required for target
2358proc macports::_deptypes_for_target {target} {
2359    switch $target {
2360        fetch       -
2361        checksum    { set deptypes "depends_fetch" }
2362        extract     -
2363        patch       { set deptypes "depends_fetch depends_extract" }
2364        configure   -
2365        build       { set deptypes "depends_fetch depends_extract depends_lib depends_build" }
2366
2367        test        -
2368        destroot    -
2369        install     -
2370        archive     -
2371        dmg         -
2372        pkg         -
2373        portpkg     -
2374        mdmg        -
2375        mpkg        -
2376        rpm         -
2377        srpm        -
2378        dpkg        -
2379        ""          { set deptypes "depends_fetch depends_extract depends_lib depends_build depends_run" }
2380    }
2381    return $deptypes
2382}
2383
2384# selfupdate procedure
2385proc macports::selfupdate {{optionslist {}} {updatestatusvar ""}} {
2386    global macports::prefix macports::portdbpath macports::libpath macports::rsync_server macports::rsync_dir macports::rsync_options
2387    global macports::autoconf::macports_version macports::autoconf::rsync_path tcl_platform
2388    array set options $optionslist
2389   
2390    # variable that indicates whether we actually updated base
2391    if {$updatestatusvar != ""} {
2392        upvar $updatestatusvar updatestatus
2393        set updatestatus no
2394    }
2395
2396    # syncing ports tree.
2397    if {![info exists options(ports_selfupdate_nosync)] || $options(ports_selfupdate_nosync) != "yes"} {
2398        ui_msg "--->  Updating the ports tree"
2399        if {[catch {mportsync $optionslist} result]} {
2400            return -code error "Couldn't sync the ports tree: $result"
2401        }
2402    }
2403
2404    # create the path to the to be downloaded sources if it doesn't exist
2405    set mp_source_path [file join $portdbpath sources ${rsync_server} ${rsync_dir}/]
2406    if {![file exists $mp_source_path]} {
2407        file mkdir $mp_source_path
2408    }
2409    ui_debug "MacPorts sources location: $mp_source_path"
2410
2411    # sync the MacPorts sources
2412    ui_msg "--->  Updating MacPorts base sources using rsync"
2413    if { [catch { system "$rsync_path $rsync_options rsync://${rsync_server}/${rsync_dir} $mp_source_path" } result ] } {
2414       return -code error "Error synchronizing MacPorts sources: $result"
2415    }
2416
2417    # echo current MacPorts version
2418    ui_msg "MacPorts base version $macports::autoconf::macports_version installed,"
2419
2420    if { [info exists options(ports_force)] && $options(ports_force) == "yes" } {
2421        set use_the_force_luke yes
2422        ui_debug "Forcing a rebuild and reinstallation of MacPorts"
2423    } else {
2424        set use_the_force_luke no
2425        ui_debug "Rebuilding and reinstalling MacPorts if needed"
2426    }
2427
2428    # Choose what version file to use: old, floating point format or new, real version number format
2429    set version_file [file join $mp_source_path config macports_version]
2430    if {[file exists $version_file]} {
2431        set fd [open $version_file r]
2432        gets $fd macports_version_new
2433        close $fd
2434        # echo downloaded MacPorts version
2435        ui_msg "MacPorts base version $macports_version_new downloaded."
2436    } else {
2437        ui_warn "No version file found, please rerun selfupdate."
2438        set macports_version_new 0
2439    }
2440
2441    # check if we we need to rebuild base
2442    set comp [rpm-vercomp $macports_version_new $macports::autoconf::macports_version]
2443    if {$use_the_force_luke == "yes" || $comp > 0} {
2444        if {[info exists options(ports_dryrun)] && $options(ports_dryrun) == "yes"} {
2445            ui_msg "--->  MacPorts base is outdated, selfupdate would install $macports_version_new (dry run)"
2446        } else {
2447            ui_msg "--->  MacPorts base is outdated, installing new version $macports_version_new"
2448
2449            # get installation user/group and permissions
2450            set owner [file attributes ${prefix} -owner]
2451            set group [file attributes ${prefix} -group]
2452            set perms [string range [file attributes ${prefix} -permissions] end-3 end]
2453            if {$tcl_platform(user) != "root" && ![string equal $tcl_platform(user) $owner]} {
2454                return -code error "User $tcl_platform(user) does not own ${prefix} - try using sudo"
2455            }
2456            ui_debug "Permissions OK"
2457
2458            # where to install our macports1.0 tcl package
2459            set mp_tclpackage_path [file join $portdbpath .tclpackage]
2460            if { [file exists $mp_tclpackage_path]} {
2461                set fd [open $mp_tclpackage_path r]
2462                gets $fd tclpackage
2463                close $fd
2464            } else {
2465                set tclpackage $libpath
2466            }
2467
2468            set configure_args "--prefix=$prefix --with-tclpackage=$tclpackage --with-install-user=$owner --with-install-group=$group --with-directory-mode=$perms"
2469            # too many users have an incompatible readline in /usr/local, see ticket #10651
2470            if {$tcl_platform(os) != "Darwin" || $prefix == "/usr/local"
2471                || ([glob -nocomplain "/usr/local/lib/lib{readline,history}*"] == "" && [glob -nocomplain "/usr/local/include/readline/*.h"] == "")} {
2472                append configure_args " --enable-readline"
2473            } else {
2474                ui_warn "Disabling readline support due to readline in /usr/local"
2475            }
2476
2477            # do the actual configure, build and installation of new base
2478            ui_msg "Installing new MacPorts release in $prefix as $owner:$group; permissions $perms; Tcl-Package in $tclpackage\n"
2479            if { [catch { system "cd $mp_source_path && ./configure $configure_args && make && make install" } result] } {
2480                return -code error "Error installing new MacPorts base: $result"
2481            }
2482            if {[info exists updatestatus]} {
2483                set updatestatus yes
2484            }
2485        }
2486    } elseif {$comp < 0} {
2487        ui_msg "--->  MacPorts base is probably trunk or a release candidate"
2488    } else {
2489        ui_msg "--->  MacPorts base is already the latest version"
2490    }
2491
2492    # set the MacPorts sources to the right owner
2493    set sources_owner [file attributes [file join $portdbpath sources/] -owner]
2494    ui_debug "Setting MacPorts sources ownership to $sources_owner"
2495    if { [catch { exec [findBinary chown $macports::autoconf::chown_path] -R $sources_owner [file join $portdbpath sources/] } result] } {
2496        return -code error "Couldn't change permissions of the MacPorts sources at $mp_source_path to $sources_owner: $result"
2497    }
2498
2499    if {![info exists options(ports_selfupdate_nosync)] || $options(ports_selfupdate_nosync) != "yes"} {
2500        ui_msg "\nThe ports tree has been updated. To upgrade your installed ports, you should run"
2501        ui_msg "  port upgrade outdated"
2502    }
2503
2504    return 0
2505}
2506
2507# upgrade API wrapper procedure
2508# return codes: 0 = success, 1 = general failure, 2 = port name not found in index
2509proc macports::upgrade {portname dspec variationslist optionslist {depscachename ""}} {
2510    # only installed ports can be upgraded
2511    if {![registry::entry_exists_for_name $portname]} {
2512        ui_error "$portname is not installed"
2513        return 1
2514    }
2515    if {![string match "" $depscachename]} {
2516        upvar $depscachename depscache
2517    } else {
2518        array set depscache {}
2519    }
2520    # stop upgrade from being called via mportexec as well
2521    set orig_nodeps yes
2522    if {![info exists macports::global_options(ports_nodeps)]} {
2523        set macports::global_options(ports_nodeps) yes
2524        set orig_nodeps no
2525    }
2526    # filter out implicit variants from the explicitly set/unset variants.
2527    set variationslist [mport_filtervariants $variationslist yes]
2528   
2529    # run the actual upgrade
2530    set status [macports::_upgrade $portname $dspec $variationslist $optionslist depscache]
2531   
2532    if {!$orig_nodeps} {
2533        unset -nocomplain macports::global_options(ports_nodeps)
2534    }
2535    return $status
2536}
2537
2538# main internal upgrade procedure
2539proc macports::_upgrade {portname dspec variationslist optionslist {depscachename ""}} {
2540    global macports::registry.installtype
2541    global macports::portarchivemode
2542    global macports::global_variations
2543    array set options $optionslist
2544
2545    # Note $variationslist is left alone and so retains the original
2546    # requested variations, which should be passed to recursive calls to
2547    # upgrade; while variations gets existing variants and global variations
2548    # merged in later on, so it applies only to this port's upgrade
2549    array set variations $variationslist
2550   
2551    set globalvarlist [array get macports::global_variations]
2552
2553    if {![string match "" $depscachename]} {
2554        upvar $depscachename depscache
2555    }
2556
2557    # Is this a dry run?
2558    set is_dryrun no
2559    if {[info exists options(ports_dryrun)] && $options(ports_dryrun) eq "yes"} {
2560        set is_dryrun yes
2561    }
2562
2563    # check if the port is in tree
2564    if {[catch {mportlookup $portname} result]} {
2565        global errorInfo
2566        ui_debug "$errorInfo"
2567        ui_error "port lookup failed: $result"
2568        return 1
2569    }
2570    # argh! port doesnt exist!
2571    if {$result == ""} {
2572        ui_warn "No port $portname found in the index."
2573        return 2
2574    }
2575    # fill array with information
2576    array set portinfo [lindex $result 1]
2577    # set portname again since the one we were passed may not have had the correct case
2578    set portname $portinfo(name)
2579
2580    # set version_in_tree and revision_in_tree
2581    if {![info exists portinfo(version)]} {
2582        ui_error "Invalid port entry for $portname, missing version"
2583        return 1
2584    }
2585    set version_in_tree "$portinfo(version)"
2586    set revision_in_tree "$portinfo(revision)"
2587    set epoch_in_tree "$portinfo(epoch)"
2588
2589    set ilist {}
2590    if { [catch {set ilist [registry::installed $portname ""]} result] } {
2591        if {$result == "Registry error: $portname not registered as installed." } {
2592            ui_debug "$portname is *not* installed by MacPorts"
2593
2594            # We need to pass _mportispresent a reference to the mport that is
2595            # actually declaring the dependency on the one we're checking for.
2596            # We got here via _upgrade_dependencies, so we grab it from 2 levels up.
2597            upvar 2 workername parentworker
2598            if {![_mportispresent $parentworker $dspec ] } {
2599                # open porthandle
2600                set porturl $portinfo(porturl)
2601                if {![info exists porturl]} {
2602                    set porturl file://./
2603                }
2604                # Merge the global variations into the specified
2605                foreach { variation value } $globalvarlist {
2606                    if { ![info exists variations($variation)] } {
2607                        set variations($variation) $value
2608                    }
2609                }
2610
2611                if {[catch {set workername [mportopen $porturl [array get options] [array get variations]]} result]} {
2612                    global errorInfo
2613                    ui_debug "$errorInfo"
2614                    ui_error "Unable to open port: $result"
2615                    return 1
2616                }
2617                # While we're at it, update the portinfo
2618                array unset portinfo
2619                array set portinfo [mportinfo $workername]
2620               
2621                # upgrade its dependencies first
2622                set status [_upgrade_dependencies portinfo depscache variationslist options yes]
2623                if {$status != 0 && ![ui_isset ports_processall]} {
2624                    catch {mportclose $workername}
2625                    return $status
2626                }
2627                # now install it
2628                if {[catch {set result [mportexec $workername install]} result]} {
2629                    global errorInfo
2630                    ui_debug "$errorInfo"
2631                    ui_error "Unable to exec port: $result"
2632                    catch {mportclose $workername}
2633                    return 1
2634                }
2635                if {$result > 0} {
2636                    ui_error "Problem while installing $portname"
2637                    catch {mportclose $workername}
2638                    return $result
2639                }
2640                # we just installed it, so mark it done in the cache
2641                set depscache(port:${portname}) 1
2642                mportclose $workername
2643            } else {
2644                # dependency is satisfied by something other than the named port
2645                ui_debug "$portname not installed, soft dependency satisfied"
2646                # mark this depspec as satisfied in the cache
2647                set depscache($dspec) 1
2648            }
2649            # the rest of the proc doesn't matter for a port that is freshly
2650            # installed or not installed
2651            return 0
2652        } else {
2653            ui_error "Checking installed version failed: $result"
2654            return 1
2655        }
2656    } else {
2657        # we'll now take care of upgrading it, so we can add it to the cache
2658        set depscache(port:${portname}) 1
2659    }
2660    set anyactive no
2661    set version_installed {}
2662    set revision_installed {}
2663    set epoch_installed 0
2664    set variant_installed ""
2665
2666    # find latest version installed and active version (if any)
2667    foreach i $ilist {
2668        set variant [lindex $i 3]
2669        set version [lindex $i 1]
2670        set revision [lindex $i 2]
2671        set epoch [lindex $i 5]
2672        if { $version_installed == {} || $epoch > $epoch_installed ||
2673                ($epoch == $epoch_installed && [rpm-vercomp $version $version_installed] > 0)
2674                || ($epoch == $epoch_installed
2675                    && [rpm-vercomp $version $version_installed] == 0
2676                    && [rpm-vercomp $revision $revision_installed] > 0)} {
2677            set version_installed $version
2678            set revision_installed $revision
2679            set variant_installed $variant
2680            set epoch_installed $epoch
2681        }
2682
2683        set isactive [lindex $i 4]
2684        if {$isactive == 1} {
2685            set anyactive yes
2686            set version_active $version
2687            set revision_active $revision
2688            set variant_active $variant
2689            set epoch_active $epoch
2690        }
2691    }
2692
2693    # output version numbers
2694    ui_debug "epoch: in tree: $epoch_in_tree installed: $epoch_installed"
2695    ui_debug "$portname ${version_in_tree}_${revision_in_tree} exists in the ports tree"
2696    ui_debug "$portname ${version_installed}_${revision_installed} $variant_installed is the latest installed"
2697    if {$anyactive} {
2698        ui_debug "$portname ${version_active}_${revision_active} $variant_active is active"
2699        # save existing variant for later use
2700        set oldvariant $variant_active
2701        set regref [registry::open_entry $portname $version_active $revision_active $variant_active $epoch_active]
2702    } else {
2703        ui_debug "no version of $portname is active"
2704        set oldvariant $variant_installed
2705        set regref [registry::open_entry $portname $version_installed $revision_installed $variant_installed $epoch_installed]
2706    }
2707    set oldnegatedvariant [registry::property_retrieve $regref negated_variants]
2708    if {$oldnegatedvariant == 0} {
2709        set oldnegatedvariant {}
2710    }
2711
2712    # Before we do
2713    # dependencies, we need to figure out the final variants,
2714    # open the port, and update the portinfo.
2715    set porturl $portinfo(porturl)
2716    if {![info exists porturl]} {
2717        set porturl file://./
2718    }
2719
2720    set minusvariant [lrange [split $oldnegatedvariant -] 1 end]
2721    set plusvariant [lrange [split $oldvariant +] 1 end]
2722    ui_debug "Merging existing variants '${oldvariant}${oldnegatedvariant}' into variants"
2723    set oldvariantlist [list]
2724    foreach v $plusvariant {
2725        lappend oldvariantlist $v "+"
2726    }
2727    foreach v $minusvariant {
2728        lappend oldvariantlist $v "-"
2729    }
2730    # remove implicit variants, without printing warnings
2731    set oldvariantlist [mport_filtervariants $oldvariantlist no]
2732
2733    # merge in the old variants
2734    foreach {variation value} $oldvariantlist {
2735        if { ![info exists variations($variation)]} {
2736            set variations($variation) $value
2737        }
2738    }
2739
2740    # Now merge in the global (i.e. variants.conf) variations.
2741    # We wait until now so that existing variants for this port
2742    # override global variations
2743    foreach { variation value } $globalvarlist {
2744        if { ![info exists variations($variation)] } {
2745            set variations($variation) $value
2746        }
2747    }
2748
2749    ui_debug "new fully merged portvariants: [array get variations]"
2750   
2751    # at this point we need to check if a different port will be replacing this one
2752    if {[info exists portinfo(replaced_by)] && ![info exists options(ports_upgrade_no-replace)]} {
2753        ui_debug "$portname is replaced by $portinfo(replaced_by)"
2754        if {[catch {mportlookup $portinfo(replaced_by)} result]} {
2755            global errorInfo
2756            ui_debug "$errorInfo"
2757            ui_error "port lookup failed: $result"
2758            return 1
2759        }
2760        if {$result == ""} {
2761            ui_error "No port $portinfo(replaced_by) found."
2762            return 1
2763        }
2764        array unset portinfo
2765        array set portinfo [lindex $result 1]
2766        set newname $portinfo(name)
2767
2768        set porturl $portinfo(porturl)
2769        if {![info exists porturl]} {
2770            set porturl file://./
2771        }
2772        set depscache(port:${newname}) 1
2773    } else {
2774        set newname $portname
2775    }
2776
2777    if {[catch {set workername [mportopen $porturl [array get options] [array get variations]]} result]} {
2778        global errorInfo
2779        ui_debug "$errorInfo"
2780        ui_error "Unable to open port: $result"
2781        return 1
2782    }
2783
2784    array unset portinfo
2785    array set portinfo [mportinfo $workername]
2786    set version_in_tree "$portinfo(version)"
2787    set revision_in_tree "$portinfo(revision)"
2788    set epoch_in_tree "$portinfo(epoch)"
2789
2790    set epoch_override 0
2791    set will_install yes
2792    # check installed version against version in ports
2793    if { ( [rpm-vercomp $version_installed $version_in_tree] > 0
2794            || ([rpm-vercomp $version_installed $version_in_tree] == 0
2795                && [rpm-vercomp $revision_installed $revision_in_tree] >= 0 ))
2796        && ![info exists options(ports_upgrade_force)] } {
2797        if {$portname != $newname} { 
2798            ui_debug "ignoring versions, installing replacement port"
2799        } elseif { $epoch_installed < $epoch_in_tree } {
2800            set epoch_override 1
2801            ui_debug "epoch override ... upgrading!"
2802        } elseif {[info exists options(ports_upgrade_enforce-variants)] && $options(ports_upgrade_enforce-variants) eq "yes"
2803                  && [info exists portinfo(canonical_active_variants)] && $portinfo(canonical_active_variants) != $oldvariant} {
2804            ui_debug "variant override ... upgrading!"
2805        } else {
2806            if {[info exists portinfo(canonical_active_variants)] && $portinfo(canonical_active_variants) != $oldvariant} {
2807                ui_warn "Skipping upgrade since $portname ${version_installed}_${revision_installed} >= $portname ${version_in_tree}_${revision_in_tree}, even though installed variants \"$oldvariant\" do not match \"$portinfo(canonical_active_variants)\". Use 'upgrade --enforce-variants' to switch to the requested variants."
2808            } else {
2809                ui_debug "No need to upgrade! $portname ${version_installed}_${revision_installed} >= $portname ${version_in_tree}_${revision_in_tree}"
2810            }
2811            set will_install no
2812        }
2813    }
2814
2815    set will_build no
2816    # avoid building again unnecessarily
2817    if {$will_install && ([info exists options(ports_upgrade_force)] || $epoch_override == 1
2818        || ![registry::entry_exists $newname $version_in_tree $revision_in_tree $portinfo(canonical_active_variants)])} {
2819        set will_build yes
2820    }
2821
2822    # first upgrade dependencies
2823    if {![info exists options(ports_nodeps)]} {
2824        set status [_upgrade_dependencies portinfo depscache variationslist options $will_build]
2825        if {$status != 0 && ![ui_isset ports_processall]} {
2826            catch {mportclose $workername}
2827            return $status
2828        }
2829    } else {
2830        ui_debug "Not following dependencies"
2831    }
2832
2833    if {!$will_install} {
2834        # nothing to do for this port, so just check if we have to do dependents
2835        if {[info exists options(ports_do_dependents)]} {
2836            # We do dependents ..
2837            set options(ports_nodeps) 1
2838
2839            registry::open_dep_map
2840            if {$anyactive} {
2841                set deplist [registry::list_dependents $portname $version_active $revision_active $variant_active]
2842            } else {
2843                set deplist [registry::list_dependents $portname $version_installed $revision_installed $variant_installed]
2844            }
2845
2846            if { [llength deplist] > 0 } {
2847                foreach dep $deplist {
2848                    set mpname [lindex $dep 2]
2849                    if {![llength [array get depscache port:${mpname}]]} {
2850                        set status [macports::_upgrade $mpname port:${mpname} $variationslist [array get options] depscache]
2851                        if {$status != 0 && ![ui_isset ports_processall]} {
2852                            catch {mportclose $workername}
2853                            return $status
2854                        }
2855                    }
2856                }
2857            }
2858        }
2859        mportclose $workername
2860        return 0
2861    }
2862
2863    if {$will_build} {
2864        # build or unarchive version_in_tree
2865        if {0 == [string compare "yes" ${macports::portarchivemode}]} {
2866            set upgrade_action "archive"
2867        } else {
2868            set upgrade_action "destroot"
2869        }
2870        if {[catch {set result [mportexec $workername $upgrade_action]} result] || $result != 0} {
2871            if {[info exists ::errorInfo]} {
2872                ui_debug "$::errorInfo"
2873            }
2874            ui_error "Unable to upgrade port: $result"
2875            catch {mportclose $workername}
2876            return 1
2877        }
2878    }
2879
2880    # always uninstall old port in direct mode
2881    if { 0 != [string compare "image" ${macports::registry.installtype}] } {
2882        # uninstall old
2883        ui_debug "Uninstalling $portname ${version_installed}_${revision_installed}${variant_installed}"
2884        # we have to force the uninstall in case of dependents
2885        set force_cur [info exists options(ports_force)]
2886        set options(ports_force) yes
2887        if {$is_dryrun eq "yes"} {
2888            ui_msg "Skipping uninstall $portname @${version_installed}_${revision_installed}${variant_installed} (dry run)"
2889        } elseif {[catch {portuninstall::uninstall $portname ${version_installed}_${revision_installed}${variant_installed} [array get options]} result]} {
2890            global errorInfo
2891            ui_debug "$errorInfo"
2892            ui_error "Uninstall $portname ${version_installed}_${revision_installed}${variant_installed} failed: $result"
2893            catch {mportclose $workername}
2894            return 1
2895        }
2896        if {!$force_cur} {
2897            unset options(ports_force)
2898        }
2899    } else {
2900        # are we installing an existing version due to force or epoch override?
2901        if {[registry::entry_exists $newname $version_in_tree $revision_in_tree $portinfo(canonical_active_variants)]
2902            && ([info exists options(ports_upgrade_force)] || $epoch_override == 1)} {
2903             ui_debug "Uninstalling $newname ${version_in_tree}_${revision_in_tree}$portinfo(canonical_active_variants)"
2904            # we have to force the uninstall in case of dependents
2905            set force_cur [info exists options(ports_force)]
2906            set options(ports_force) yes
2907            if {$is_dryrun eq "yes"} {
2908                ui_msg "Skipping uninstall $newname @${version_in_tree}_${revision_in_tree}$portinfo(canonical_active_variants) (dry run)"
2909            } elseif {[catch {portuninstall::uninstall $newname ${version_in_tree}_${revision_in_tree}$portinfo(canonical_active_variants) [array get options]} result]} {
2910                global errorInfo
2911                ui_debug "$errorInfo"
2912                ui_error "Uninstall $newname ${version_in_tree}_${revision_in_tree}$portinfo(canonical_active_variants) failed: $result"
2913                catch {mportclose $workername}
2914                return 1
2915            }
2916            if {!$force_cur} {
2917                unset options(ports_force)
2918            }
2919            if {$anyactive && $version_in_tree == $version_active && $revision_in_tree == $revision_active
2920                && $portinfo(canonical_active_variants) == $variant_active && $portname == $newname} {
2921                set anyactive no
2922            }
2923        }
2924        if {[info exists options(port_uninstall_old)]} {
2925            # uninstalling now could fail due to dependents when not forced,
2926            # because the new version is not installed
2927            set uninstall_later yes
2928        }
2929    }
2930
2931    if {$is_dryrun eq "yes"} {
2932        if {$anyactive} {
2933            ui_msg "Skipping deactivate $portname @${version_active}_${revision_active}${variant_active} (dry run)"
2934        }
2935        ui_msg "Skipping activate $newname @${version_in_tree}_${revision_in_tree}$portinfo(canonical_active_variants) (dry run)"
2936    } elseif {[catch {set result [mportexec $workername install]} result]} {
2937        global errorInfo
2938        ui_debug "$errorInfo"
2939        ui_error "Couldn't activate $newname ${version_in_tree}_${revision_in_tree}$portinfo(canonical_active_variants): $result"
2940        catch {mportclose $workername}
2941        return 1
2942    }
2943
2944    # Check if we have to do dependents
2945    if {[info exists options(ports_do_dependents)]} {
2946        # We do dependents ..
2947        set options(ports_nodeps) 1
2948
2949        registry::open_dep_map
2950        if {$portname != $newname} {
2951            set deplist [registry::list_dependents $newname $version_in_tree $revision_in_tree $portinfo(canonical_active_variants)]
2952        } else {
2953            set deplist [list]
2954        }
2955        if {$anyactive} {
2956            set deplist [concat $deplist [registry::list_dependents $portname $version_active $revision_active $variant_active]]
2957        } else {
2958            set deplist [concat $deplist [registry::list_dependents $portname $version_installed $revision_installed $variant_installed]]
2959        }
2960
2961        if { [llength deplist] > 0 } {
2962            foreach dep $deplist {
2963                set mpname [lindex $dep 2]
2964                if {![llength [array get depscache port:${mpname}]]} {
2965                    set status [macports::_upgrade $mpname port:${mpname} $variationslist [array get options] depscache]
2966                    if {$status != 0 && ![ui_isset ports_processall]} {
2967                        catch {mportclose $workername}
2968                        return $status
2969                    }
2970                }
2971            }
2972        }
2973    }
2974
2975    if {[info exists uninstall_later] && $uninstall_later == yes} {
2976        foreach i $ilist {
2977            set version [lindex $i 1]
2978            set revision [lindex $i 2]
2979            set variant [lindex $i 3]
2980            if {$version == $version_in_tree && $revision == $revision_in_tree && $variant == $portinfo(canonical_active_variants) && $portname == $newname} {
2981                continue
2982            }
2983            ui_debug "Uninstalling $portname ${version}_${revision}${variant}"
2984            if {$is_dryrun eq "yes"} {
2985                ui_msg "Skipping uninstall $portname @${version}_${revision}${variant} (dry run)"
2986            } elseif {[catch {portuninstall::uninstall $portname ${version}_${revision}${variant} $optionslist} result]} {
2987                global errorInfo
2988                ui_debug "$errorInfo"
2989                # replaced_by can mean that we try to uninstall all versions of the old port, so handle errors due to dependents
2990                if {$result != "Please uninstall the ports that depend on $portname first." && ![ui_isset ports_processall]} {
2991                    ui_error "Uninstall $portname @${version}_${revision}${variant} failed: $result"
2992                    catch {mportclose $workername}
2993                    return 1
2994                }
2995            }
2996        }
2997    }
2998
2999    # close the port handle
3000    mportclose $workername
3001    return 0
3002}
3003
3004# upgrade_dependencies: helper proc for upgrade
3005# Calls upgrade on each dependency listed in the PortInfo.
3006# Uses upvar to access the variables.
3007proc macports::_upgrade_dependencies {portinfoname depscachename variationslistname optionsname {build_needed yes}} {
3008    upvar $portinfoname portinfo $depscachename depscache \
3009          $variationslistname variationslist \
3010          $optionsname options
3011    upvar workername parentworker
3012
3013    # If we're following dependents, we only want to follow this port's
3014    # dependents, not those of all its dependencies. Otherwise, we would
3015    # end up processing this port's dependents n+1 times (recursively!),
3016    # where n is the number of dependencies this port has, since this port
3017    # is of course a dependent of each of its dependencies. Plus the
3018    # dependencies could have any number of unrelated dependents.
3019
3020    # So we save whether we're following dependents, unset the option
3021    # while doing the dependencies, and restore it afterwards.
3022    set saved_do_dependents [info exists options(ports_do_dependents)]
3023    unset -nocomplain options(ports_do_dependents)
3024
3025    set status 0
3026    # each required dep type is upgraded
3027    if {$build_needed} {
3028        set dtypes {depends_fetch depends_extract depends_build depends_lib depends_run}
3029    } else {
3030        set dtypes {depends_lib depends_run}
3031    }
3032    foreach dtype $dtypes {
3033        if {[info exists portinfo($dtype)]} {
3034            foreach i $portinfo($dtype) {
3035                set parent_interp [ditem_key $parentworker workername]
3036                set d [$parent_interp eval _get_dep_port $i]
3037                if {![llength [array get depscache port:${d}]] && ![llength [array get depscache $i]]} {
3038                    if {$d != ""} {
3039                        set dspec port:$d
3040                    } else {
3041                        set dspec $i
3042                        set d [lindex [split $i :] end]
3043                    }
3044                    set status [macports::_upgrade $d $dspec $variationslist [array get options] depscache]
3045                    if {$status != 0 && ![ui_isset ports_processall]} break
3046                }
3047            }
3048        }
3049        if {$status != 0 && ![ui_isset ports_processall]} break
3050    }
3051    # restore dependent-following to its former value
3052    if {$saved_do_dependents} {
3053        set options(ports_do_dependents) yes
3054    }
3055    return $status
3056}
3057
3058# mportselect
3059#   * command: The only valid commands are list, set and show
3060#   * group: This argument should correspond to a directory under
3061#            $macports::prefix/etc/select.
3062#   * version: This argument is only used by the 'set' command.
3063# On error mportselect returns with the code 'error'.
3064proc mportselect {command group {version ""}} {
3065    ui_debug "mportselect \[$command] \[$group] \[$version]"
3066
3067    set conf_path "$macports::prefix/etc/select/$group"
3068    if {![file isdirectory $conf_path]} {
3069        return -code error "The specified group '$group' does not exist."
3070    }
3071
3072    switch -- $command {
3073        list {
3074            if {[catch {set versions [glob -directory $conf_path *]}]} {
3075                return -code error [concat "No configurations associated " \
3076                                           "with '$group' were found."]
3077            }
3078
3079            # Return the sorted list of versions (excluding base and current).
3080            set lversions {}
3081            foreach v $versions {
3082                # Only the file name corresponds to the version name.
3083                set v [file tail $v]
3084                if {$v eq "base" || $v eq "current"} {
3085                    continue
3086                }
3087                lappend lversions [file tail $v]
3088            }
3089            return [lsort $lversions]
3090        }
3091        set {
3092            # Use $conf_path/$version to read in sources.
3093            if {[catch {set src_file [open "$conf_path/$version"]}]} {
3094                return -code error [concat "Verify that the specified " \
3095                                           "version '$version' is valid " \
3096                                           "(i.e., Is it listed when you " \
3097                                           "specify the --list command?)."]
3098            }
3099            set srcs [split [read -nonewline $src_file] "\n"]
3100            close $src_file
3101
3102            # Use $conf_path/base to read in targets.
3103            if {[catch {set tgt_file [open "$conf_path/base"]}]} {
3104                return -code error [concat "The configuration file " \
3105                                           "'$conf_path/base' could not be " \
3106                                           "opened."]
3107            }
3108            set tgts [split [read -nonewline $tgt_file] "\n"]
3109            close $tgt_file
3110
3111            # Iterate through the configuration files executing the specified
3112            # actions.
3113            set i 0
3114            foreach tgt $tgts {
3115                set src [lindex $srcs $i]
3116
3117                switch -glob -- $src {
3118                    - {
3119                        # The source is unavailable for this file.
3120                        set tgt [file join $macports::prefix $tgt]
3121                        file delete $tgt
3122                        ui_debug "rm -f $tgt"
3123                    }
3124                    /* {
3125                        # The source is an absolute path.
3126                        set tgt [file join $macports::prefix $tgt]
3127                        file delete $tgt
3128                        file link -symbolic $tgt $src
3129                        ui_debug "ln -sf $src $tgt"
3130                    }
3131                    default {
3132                        # The source is a relative path.
3133                        set src [file join $macports::prefix $src]
3134                        set tgt [file join $macports::prefix $tgt]
3135                        file delete $tgt
3136                        file link -symbolic $tgt $src
3137                        ui_debug "ln -sf $src $tgt"
3138                    }
3139                }
3140                set i [expr $i+1]
3141            }
3142
3143            # Update the selected version.
3144            set selected_version "$conf_path/current"
3145            if {[file exists $selected_version]} {
3146                file delete $selected_version
3147            }
3148            symlink $version $selected_version
3149            return
3150        }
3151        show {
3152            set selected_version "$conf_path/current"
3153
3154            if {![file exists $selected_version]} {
3155                return "none"
3156            } else {
3157                return [file readlink $selected_version]
3158            }
3159        }
3160    }
3161    return
3162}
Note: See TracBrowser for help on using the repository browser.