Martin v. Löwis | 8ffe9ab | 2004-08-22 13:34:34 +0000 | [diff] [blame] | 1 | # Python MSI Generator |
| 2 | # (C) 2003 Martin v. Loewis |
| 3 | # See "FOO" in comments refers to MSDN sections with the title FOO. |
Martin v. Löwis | 9fda931 | 2004-12-22 13:41:49 +0000 | [diff] [blame] | 4 | import msilib, schema, sequence, os, glob, time, re |
Martin v. Löwis | 8ffe9ab | 2004-08-22 13:34:34 +0000 | [diff] [blame] | 5 | from msilib import Feature, CAB, Directory, Dialog, Binary, add_data |
| 6 | import uisample |
| 7 | from win32com.client import constants |
Martin v. Löwis | 9fda931 | 2004-12-22 13:41:49 +0000 | [diff] [blame] | 8 | from distutils.spawn import find_executable |
Martin v. Löwis | 1d278fc | 2006-03-28 18:30:05 +0000 | [diff] [blame] | 9 | from uuids import product_codes |
Martin v. Löwis | 8ffe9ab | 2004-08-22 13:34:34 +0000 | [diff] [blame] | 10 | |
| 11 | # Settings can be overridden in config.py below |
Martin v. Löwis | 8ffe9ab | 2004-08-22 13:34:34 +0000 | [diff] [blame] | 12 | # 0 for official python.org releases |
| 13 | # 1 for intermediate releases by anybody, with |
| 14 | # a new product code for every package. |
| 15 | snapshot = 1 |
| 16 | # 1 means that file extension is px, not py, |
| 17 | # and binaries start with x |
| 18 | testpackage = 0 |
| 19 | # Location of build tree |
| 20 | srcdir = os.path.abspath("../..") |
| 21 | # Text to be displayed as the version in dialogs etc. |
| 22 | # goes into file name and ProductCode. Defaults to |
| 23 | # current_version.day for Snapshot, current_version otherwise |
| 24 | full_current_version = None |
Martin v. Löwis | e0f780d | 2004-09-01 14:51:06 +0000 | [diff] [blame] | 25 | # Is Tcl available at all? |
| 26 | have_tcl = True |
Martin v. Löwis | 1a494bd | 2006-04-04 07:10:59 +0000 | [diff] [blame] | 27 | # Where is sqlite3.dll located, relative to srcdir? |
| 28 | sqlite_dir = "../sqlite-source-3.3.4" |
Martin v. Löwis | 8ffe9ab | 2004-08-22 13:34:34 +0000 | [diff] [blame] | 29 | |
| 30 | try: |
| 31 | from config import * |
| 32 | except ImportError: |
| 33 | pass |
| 34 | |
| 35 | # Extract current version from Include/patchlevel.h |
| 36 | lines = open(srcdir + "/Include/patchlevel.h").readlines() |
| 37 | major = minor = micro = level = serial = None |
| 38 | levels = { |
| 39 | 'PY_RELEASE_LEVEL_ALPHA':0xA, |
| 40 | 'PY_RELEASE_LEVEL_BETA': 0xB, |
| 41 | 'PY_RELEASE_LEVEL_GAMMA':0xC, |
| 42 | 'PY_RELEASE_LEVEL_FINAL':0xF |
| 43 | } |
| 44 | for l in lines: |
| 45 | if not l.startswith("#define"): |
| 46 | continue |
| 47 | l = l.split() |
| 48 | if len(l) != 3: |
| 49 | continue |
| 50 | _, name, value = l |
| 51 | if name == 'PY_MAJOR_VERSION': major = value |
| 52 | if name == 'PY_MINOR_VERSION': minor = value |
| 53 | if name == 'PY_MICRO_VERSION': micro = value |
| 54 | if name == 'PY_RELEASE_LEVEL': level = levels[value] |
| 55 | if name == 'PY_RELEASE_SERIAL': serial = value |
| 56 | |
| 57 | short_version = major+"."+minor |
| 58 | # See PC/make_versioninfo.c |
| 59 | FIELD3 = 1000*int(micro) + 10*level + int(serial) |
| 60 | current_version = "%s.%d" % (short_version, FIELD3) |
| 61 | |
| 62 | # This should never change. The UpgradeCode of this package can be |
| 63 | # used in the Upgrade table of future packages to make the future |
| 64 | # package replace this one. See "UpgradeCode Property". |
| 65 | upgrade_code_snapshot='{92A24481-3ECB-40FC-8836-04B7966EC0D5}' |
| 66 | upgrade_code='{65E6DE48-A358-434D-AA4F-4AF72DB4718F}' |
| 67 | |
Martin v. Löwis | 8ffe9ab | 2004-08-22 13:34:34 +0000 | [diff] [blame] | 68 | if snapshot: |
| 69 | current_version = "%s.%s.%s" % (major, minor, int(time.time()/3600/24)) |
| 70 | product_code = msilib.gen_uuid() |
| 71 | else: |
| 72 | product_code = product_codes[current_version] |
| 73 | |
| 74 | if full_current_version is None: |
| 75 | full_current_version = current_version |
| 76 | |
| 77 | extensions = [ |
| 78 | 'bz2.pyd', |
| 79 | 'pyexpat.pyd', |
| 80 | 'select.pyd', |
| 81 | 'unicodedata.pyd', |
| 82 | 'winsound.pyd', |
Trent Mick | e97e5a7 | 2005-12-15 22:08:46 +0000 | [diff] [blame] | 83 | '_elementtree.pyd', |
Martin v. Löwis | 8ffe9ab | 2004-08-22 13:34:34 +0000 | [diff] [blame] | 84 | '_bsddb.pyd', |
| 85 | '_socket.pyd', |
| 86 | '_ssl.pyd', |
| 87 | '_testcapi.pyd', |
| 88 | '_tkinter.pyd', |
Martin v. Löwis | 8c7c56e | 2006-03-05 14:04:26 +0000 | [diff] [blame] | 89 | '_msi.pyd', |
Martin v. Löwis | a09655e | 2006-03-10 15:36:28 +0000 | [diff] [blame] | 90 | '_ctypes.pyd', |
Martin v. Löwis | 1a494bd | 2006-04-04 07:10:59 +0000 | [diff] [blame] | 91 | '_ctypes_test.pyd', |
| 92 | '_sqlite3.pyd' |
Martin v. Löwis | 8ffe9ab | 2004-08-22 13:34:34 +0000 | [diff] [blame] | 93 | ] |
| 94 | |
Martin v. Löwis | 141f41a | 2005-03-15 00:39:40 +0000 | [diff] [blame] | 95 | # Well-known component UUIDs |
| 96 | # These are needed for SharedDLLs reference counter; if |
| 97 | # a different UUID was used for each incarnation of, say, |
| 98 | # python24.dll, an upgrade would set the reference counter |
| 99 | # from 1 to 2 (due to what I consider a bug in MSI) |
| 100 | # Using the same UUID is fine since these files are versioned, |
| 101 | # so Installer will always keep the newest version. |
| 102 | msvcr71_uuid = "{8666C8DD-D0B4-4B42-928E-A69E32FA5D4D}" |
| 103 | pythondll_uuid = { |
| 104 | "24":"{9B81E618-2301-4035-AC77-75D9ABEB7301}", |
| 105 | "25":"{2e41b118-38bd-4c1b-a840-6977efd1b911}" |
| 106 | } [major+minor] |
Tim Peters | eba28be | 2005-03-28 01:08:02 +0000 | [diff] [blame] | 107 | |
Martin v. Löwis | 9fda931 | 2004-12-22 13:41:49 +0000 | [diff] [blame] | 108 | # Build the mingw import library, libpythonXY.a |
| 109 | # This requires 'nm' and 'dlltool' executables on your PATH |
| 110 | def build_mingw_lib(lib_file, def_file, dll_file, mingw_lib): |
| 111 | warning = "WARNING: %s - libpythonXX.a not built" |
| 112 | nm = find_executable('nm') |
| 113 | dlltool = find_executable('dlltool') |
| 114 | |
| 115 | if not nm or not dlltool: |
| 116 | print warning % "nm and/or dlltool were not found" |
| 117 | return False |
| 118 | |
| 119 | nm_command = '%s -Cs %s' % (nm, lib_file) |
| 120 | dlltool_command = "%s --dllname %s --def %s --output-lib %s" % \ |
| 121 | (dlltool, dll_file, def_file, mingw_lib) |
| 122 | export_match = re.compile(r"^_imp__(.*) in python\d+\.dll").match |
| 123 | |
| 124 | f = open(def_file,'w') |
| 125 | print >>f, "LIBRARY %s" % dll_file |
| 126 | print >>f, "EXPORTS" |
| 127 | |
| 128 | nm_pipe = os.popen(nm_command) |
| 129 | for line in nm_pipe.readlines(): |
| 130 | m = export_match(line) |
| 131 | if m: |
| 132 | print >>f, m.group(1) |
| 133 | f.close() |
| 134 | exit = nm_pipe.close() |
| 135 | |
| 136 | if exit: |
| 137 | print warning % "nm did not run successfully" |
| 138 | return False |
| 139 | |
| 140 | if os.system(dlltool_command) != 0: |
| 141 | print warning % "dlltool did not run successfully" |
| 142 | return False |
| 143 | |
| 144 | return True |
| 145 | |
| 146 | # Target files (.def and .a) go in PCBuild directory |
| 147 | lib_file = os.path.join(srcdir, "PCBuild", "python%s%s.lib" % (major, minor)) |
| 148 | def_file = os.path.join(srcdir, "PCBuild", "python%s%s.def" % (major, minor)) |
| 149 | dll_file = "python%s%s.dll" % (major, minor) |
| 150 | mingw_lib = os.path.join(srcdir, "PCBuild", "libpython%s%s.a" % (major, minor)) |
| 151 | |
| 152 | have_mingw = build_mingw_lib(lib_file, def_file, dll_file, mingw_lib) |
| 153 | |
Martin v. Löwis | 856bf9a | 2006-02-14 20:42:55 +0000 | [diff] [blame] | 154 | # Determine the target architechture |
| 155 | dll_path = os.path.join(srcdir, "PCBuild", dll_file) |
| 156 | msilib.set_arch_from_file(dll_path) |
| 157 | if msilib.pe_type(dll_path) != msilib.pe_type("msisupport.dll"): |
| 158 | raise SystemError, "msisupport.dll for incorrect architecture" |
| 159 | |
Martin v. Löwis | 8ffe9ab | 2004-08-22 13:34:34 +0000 | [diff] [blame] | 160 | if testpackage: |
| 161 | ext = 'px' |
| 162 | testprefix = 'x' |
| 163 | else: |
| 164 | ext = 'py' |
| 165 | testprefix = '' |
| 166 | |
| 167 | if msilib.Win64: |
| 168 | SystemFolderName = "[SystemFolder64]" |
| 169 | else: |
| 170 | SystemFolderName = "[SystemFolder]" |
| 171 | |
| 172 | msilib.reset() |
| 173 | |
| 174 | # condition in which to install pythonxy.dll in system32: |
| 175 | # a) it is Windows 9x or |
| 176 | # b) it is NT, the user is privileged, and has chosen per-machine installation |
| 177 | sys32cond = "(Windows9x or (Privileged and ALLUSERS))" |
| 178 | |
Martin v. Löwis | 8ffe9ab | 2004-08-22 13:34:34 +0000 | [diff] [blame] | 179 | def build_database(): |
| 180 | """Generate an empty database, with just the schema and the |
| 181 | Summary information stream.""" |
| 182 | if snapshot: |
| 183 | uc = upgrade_code_snapshot |
| 184 | else: |
| 185 | uc = upgrade_code |
| 186 | # schema represents the installer 2.0 database schema. |
| 187 | # sequence is the set of standard sequences |
| 188 | # (ui/execute, admin/advt/install) |
Martin v. Löwis | 856bf9a | 2006-02-14 20:42:55 +0000 | [diff] [blame] | 189 | db = msilib.init_database("python-%s%s.msi" % (full_current_version, msilib.arch_ext), |
Martin v. Löwis | 8ffe9ab | 2004-08-22 13:34:34 +0000 | [diff] [blame] | 190 | schema, ProductName="Python "+full_current_version, |
| 191 | ProductCode=product_code, |
| 192 | ProductVersion=current_version, |
| 193 | Manufacturer=u"Martin v. L\xf6wis") |
| 194 | # The default sequencing of the RemoveExistingProducts action causes |
| 195 | # removal of files that got just installed. Place it after |
| 196 | # InstallInitialize, so we first uninstall everything, but still roll |
| 197 | # back in case the installation is interrupted |
| 198 | msilib.change_sequence(sequence.InstallExecuteSequence, |
| 199 | "RemoveExistingProducts", 1510) |
Martin v. Löwis | 8ffe9ab | 2004-08-22 13:34:34 +0000 | [diff] [blame] | 200 | msilib.add_tables(db, sequence) |
| 201 | # We cannot set ALLUSERS in the property table, as this cannot be |
| 202 | # reset if the user choses a per-user installation. Instead, we |
| 203 | # maintain WhichUsers, which can be "ALL" or "JUSTME". The UI manages |
| 204 | # this property, and when the execution starts, ALLUSERS is set |
| 205 | # accordingly. |
| 206 | add_data(db, "Property", [("UpgradeCode", uc), |
| 207 | ("WhichUsers", "ALL"), |
Martin v. Löwis | 141f41a | 2005-03-15 00:39:40 +0000 | [diff] [blame] | 208 | ("ProductLine", "Python%s%s" % (major, minor)), |
Martin v. Löwis | 8ffe9ab | 2004-08-22 13:34:34 +0000 | [diff] [blame] | 209 | ]) |
| 210 | db.Commit() |
| 211 | return db |
| 212 | |
| 213 | def remove_old_versions(db): |
| 214 | "Fill the upgrade table." |
| 215 | start = "%s.%s.0" % (major, minor) |
| 216 | # This requests that feature selection states of an older |
| 217 | # installation should be forwarded into this one. Upgrading |
| 218 | # requires that both the old and the new installation are |
| 219 | # either both per-machine or per-user. |
| 220 | migrate_features = 1 |
| 221 | # See "Upgrade Table". We remove releases with the same major and |
| 222 | # minor version. For an snapshot, we remove all earlier snapshots. For |
| 223 | # a release, we remove all snapshots, and all earlier releases. |
| 224 | if snapshot: |
| 225 | add_data(db, "Upgrade", |
Tim Peters | 66cb018 | 2004-08-26 05:23:19 +0000 | [diff] [blame] | 226 | [(upgrade_code_snapshot, start, |
Martin v. Löwis | 8ffe9ab | 2004-08-22 13:34:34 +0000 | [diff] [blame] | 227 | current_version, |
| 228 | None, # Ignore language |
Tim Peters | 66cb018 | 2004-08-26 05:23:19 +0000 | [diff] [blame] | 229 | migrate_features, |
Martin v. Löwis | 8ffe9ab | 2004-08-22 13:34:34 +0000 | [diff] [blame] | 230 | None, # Migrate ALL features |
| 231 | "REMOVEOLDSNAPSHOT")]) |
| 232 | props = "REMOVEOLDSNAPSHOT" |
| 233 | else: |
| 234 | add_data(db, "Upgrade", |
| 235 | [(upgrade_code, start, current_version, |
| 236 | None, migrate_features, None, "REMOVEOLDVERSION"), |
| 237 | (upgrade_code_snapshot, start, "%s.%d.0" % (major, int(minor)+1), |
| 238 | None, migrate_features, None, "REMOVEOLDSNAPSHOT")]) |
| 239 | props = "REMOVEOLDSNAPSHOT;REMOVEOLDVERSION" |
| 240 | # Installer collects the product codes of the earlier releases in |
| 241 | # these properties. In order to allow modification of the properties, |
| 242 | # they must be declared as secure. See "SecureCustomProperties Property" |
| 243 | add_data(db, "Property", [("SecureCustomProperties", props)]) |
| 244 | |
| 245 | class PyDialog(Dialog): |
| 246 | """Dialog class with a fixed layout: controls at the top, then a ruler, |
| 247 | then a list of buttons: back, next, cancel. Optionally a bitmap at the |
| 248 | left.""" |
| 249 | def __init__(self, *args, **kw): |
| 250 | """Dialog(database, name, x, y, w, h, attributes, title, first, |
| 251 | default, cancel, bitmap=true)""" |
| 252 | Dialog.__init__(self, *args) |
| 253 | ruler = self.h - 36 |
| 254 | bmwidth = 152*ruler/328 |
| 255 | if kw.get("bitmap", True): |
| 256 | self.bitmap("Bitmap", 0, 0, bmwidth, ruler, "PythonWin") |
| 257 | self.line("BottomLine", 0, ruler, self.w, 0) |
| 258 | |
| 259 | def title(self, title): |
| 260 | "Set the title text of the dialog at the top." |
| 261 | # name, x, y, w, h, flags=Visible|Enabled|Transparent|NoPrefix, |
| 262 | # text, in VerdanaBold10 |
| 263 | self.text("Title", 135, 10, 220, 60, 0x30003, |
| 264 | r"{\VerdanaBold10}%s" % title) |
| 265 | |
| 266 | def back(self, title, next, name = "Back", active = 1): |
| 267 | """Add a back button with a given title, the tab-next button, |
| 268 | its name in the Control table, possibly initially disabled. |
| 269 | |
| 270 | Return the button, so that events can be associated""" |
| 271 | if active: |
| 272 | flags = 3 # Visible|Enabled |
| 273 | else: |
| 274 | flags = 1 # Visible |
| 275 | return self.pushbutton(name, 180, self.h-27 , 56, 17, flags, title, next) |
| 276 | |
| 277 | def cancel(self, title, next, name = "Cancel", active = 1): |
| 278 | """Add a cancel button with a given title, the tab-next button, |
| 279 | its name in the Control table, possibly initially disabled. |
| 280 | |
| 281 | Return the button, so that events can be associated""" |
| 282 | if active: |
| 283 | flags = 3 # Visible|Enabled |
| 284 | else: |
| 285 | flags = 1 # Visible |
| 286 | return self.pushbutton(name, 304, self.h-27, 56, 17, flags, title, next) |
| 287 | |
| 288 | def next(self, title, next, name = "Next", active = 1): |
| 289 | """Add a Next button with a given title, the tab-next button, |
| 290 | its name in the Control table, possibly initially disabled. |
| 291 | |
| 292 | Return the button, so that events can be associated""" |
| 293 | if active: |
| 294 | flags = 3 # Visible|Enabled |
| 295 | else: |
| 296 | flags = 1 # Visible |
| 297 | return self.pushbutton(name, 236, self.h-27, 56, 17, flags, title, next) |
| 298 | |
| 299 | def xbutton(self, name, title, next, xpos): |
| 300 | """Add a button with a given title, the tab-next button, |
| 301 | its name in the Control table, giving its x position; the |
| 302 | y-position is aligned with the other buttons. |
| 303 | |
| 304 | Return the button, so that events can be associated""" |
| 305 | return self.pushbutton(name, int(self.w*xpos - 28), self.h-27, 56, 17, 3, title, next) |
| 306 | |
| 307 | def add_ui(db): |
| 308 | x = y = 50 |
| 309 | w = 370 |
| 310 | h = 300 |
| 311 | title = "[ProductName] Setup" |
| 312 | |
| 313 | # see "Dialog Style Bits" |
| 314 | modal = 3 # visible | modal |
| 315 | modeless = 1 # visible |
| 316 | track_disk_space = 32 |
| 317 | |
| 318 | add_data(db, 'ActionText', uisample.ActionText) |
| 319 | add_data(db, 'UIText', uisample.UIText) |
| 320 | |
| 321 | # Bitmaps |
| 322 | if not os.path.exists(srcdir+r"\PC\python_icon.exe"): |
| 323 | raise "Run icons.mak in PC directory" |
| 324 | add_data(db, "Binary", |
| 325 | [("PythonWin", msilib.Binary(srcdir+r"\PCbuild\installer.bmp")), # 152x328 pixels |
| 326 | ("py.ico",msilib.Binary(srcdir+r"\PC\py.ico")), |
| 327 | ]) |
| 328 | add_data(db, "Icon", |
| 329 | [("python_icon.exe", msilib.Binary(srcdir+r"\PC\python_icon.exe"))]) |
| 330 | |
| 331 | # Scripts |
Martin v. Löwis | dff68d0 | 2004-09-10 09:20:10 +0000 | [diff] [blame] | 332 | # CheckDir sets TargetExists if TARGETDIR exists. |
| 333 | # UpdateEditIDLE sets the REGISTRY.tcl component into |
| 334 | # the installed/uninstalled state according to both the |
| 335 | # Extensions and TclTk features. |
Martin v. Löwis | eb68be4 | 2004-12-12 15:29:21 +0000 | [diff] [blame] | 336 | if os.system("nmake /nologo /c /f msisupport.mak") != 0: |
| 337 | raise "'nmake /f msisupport.mak' failed" |
| 338 | add_data(db, "Binary", [("Script", msilib.Binary("msisupport.dll"))]) |
| 339 | # See "Custom Action Type 1" |
Martin v. Löwis | 3390d33 | 2005-03-14 17:20:13 +0000 | [diff] [blame] | 340 | if msilib.Win64: |
| 341 | CheckDir = "CheckDir" |
Martin v. Löwis | df40ce3 | 2006-02-16 14:38:30 +0000 | [diff] [blame] | 342 | UpdateEditIDLE = "UpdateEditIDLE" |
Martin v. Löwis | 3390d33 | 2005-03-14 17:20:13 +0000 | [diff] [blame] | 343 | else: |
| 344 | CheckDir = "_CheckDir@4" |
| 345 | UpdateEditIDLE = "_UpdateEditIDLE@4" |
Tim Peters | 0e9980f | 2004-09-12 03:49:31 +0000 | [diff] [blame] | 346 | add_data(db, "CustomAction", |
Martin v. Löwis | 3390d33 | 2005-03-14 17:20:13 +0000 | [diff] [blame] | 347 | [("CheckDir", 1, "Script", CheckDir)]) |
Martin v. Löwis | eac02e6 | 2004-11-18 08:00:33 +0000 | [diff] [blame] | 348 | if have_tcl: |
| 349 | add_data(db, "CustomAction", |
Martin v. Löwis | 3390d33 | 2005-03-14 17:20:13 +0000 | [diff] [blame] | 350 | [("UpdateEditIDLE", 1, "Script", UpdateEditIDLE)]) |
Martin v. Löwis | 8ffe9ab | 2004-08-22 13:34:34 +0000 | [diff] [blame] | 351 | |
| 352 | # UI customization properties |
| 353 | add_data(db, "Property", |
| 354 | # See "DefaultUIFont Property" |
| 355 | [("DefaultUIFont", "DlgFont8"), |
| 356 | # See "ErrorDialog Style Bit" |
| 357 | ("ErrorDialog", "ErrorDlg"), |
| 358 | ("Progress1", "Install"), # modified in maintenance type dlg |
| 359 | ("Progress2", "installs"), |
| 360 | ("MaintenanceForm_Action", "Repair")]) |
| 361 | |
| 362 | # Fonts, see "TextStyle Table" |
| 363 | add_data(db, "TextStyle", |
| 364 | [("DlgFont8", "Tahoma", 9, None, 0), |
| 365 | ("DlgFontBold8", "Tahoma", 8, None, 1), #bold |
| 366 | ("VerdanaBold10", "Verdana", 10, None, 1), |
Martin v. Löwis | 141f41a | 2005-03-15 00:39:40 +0000 | [diff] [blame] | 367 | ("VerdanaRed9", "Verdana", 9, 255, 0), |
Martin v. Löwis | 8ffe9ab | 2004-08-22 13:34:34 +0000 | [diff] [blame] | 368 | ]) |
| 369 | |
Martin v. Löwis | 9fbc44c | 2006-04-07 10:02:25 +0000 | [diff] [blame] | 370 | compileargs = r"-Wi [TARGETDIR]Lib\compileall.py -f -x bad_coding|badsyntax|site-packages [TARGETDIR]Lib" |
Martin v. Löwis | 8ffe9ab | 2004-08-22 13:34:34 +0000 | [diff] [blame] | 371 | # See "CustomAction Table" |
| 372 | add_data(db, "CustomAction", [ |
| 373 | # msidbCustomActionTypeFirstSequence + msidbCustomActionTypeTextData + msidbCustomActionTypeProperty |
| 374 | # See "Custom Action Type 51", |
| 375 | # "Custom Action Execution Scheduling Options" |
| 376 | ("InitialTargetDir", 307, "TARGETDIR", |
| 377 | "[WindowsVolume]Python%s%s" % (major, minor)), |
| 378 | ("SetDLLDirToTarget", 307, "DLLDIR", "[TARGETDIR]"), |
| 379 | ("SetDLLDirToSystem32", 307, "DLLDIR", SystemFolderName), |
| 380 | # msidbCustomActionTypeExe + msidbCustomActionTypeSourceFile |
| 381 | # See "Custom Action Type 18" |
Martin v. Löwis | 7b2563b | 2004-11-02 22:59:56 +0000 | [diff] [blame] | 382 | ("CompilePyc", 18, "python.exe", compileargs), |
| 383 | ("CompilePyo", 18, "python.exe", "-O "+compileargs), |
Martin v. Löwis | 8ffe9ab | 2004-08-22 13:34:34 +0000 | [diff] [blame] | 384 | ]) |
| 385 | |
| 386 | # UI Sequences, see "InstallUISequence Table", "Using a Sequence Table" |
| 387 | # Numbers indicate sequence; see sequence.py for how these action integrate |
| 388 | add_data(db, "InstallUISequence", |
| 389 | [("PrepareDlg", "Not Privileged or Windows9x or Installed", 140), |
| 390 | ("WhichUsersDlg", "Privileged and not Windows9x and not Installed", 141), |
| 391 | ("InitialTargetDir", 'TARGETDIR=""', 750), |
| 392 | # In the user interface, assume all-users installation if privileged. |
| 393 | ("SetDLLDirToSystem32", 'DLLDIR="" and ' + sys32cond, 751), |
| 394 | ("SetDLLDirToTarget", 'DLLDIR="" and not ' + sys32cond, 752), |
| 395 | ("SelectDirectoryDlg", "Not Installed", 1230), |
| 396 | # XXX no support for resume installations yet |
| 397 | #("ResumeDlg", "Installed AND (RESUME OR Preselected)", 1240), |
| 398 | ("MaintenanceTypeDlg", "Installed AND NOT RESUME AND NOT Preselected", 1250), |
| 399 | ("ProgressDlg", None, 1280)]) |
| 400 | add_data(db, "AdminUISequence", |
| 401 | [("InitialTargetDir", 'TARGETDIR=""', 750), |
| 402 | ("SetDLLDirToTarget", 'DLLDIR=""', 751), |
| 403 | ]) |
| 404 | |
| 405 | # Execute Sequences |
| 406 | add_data(db, "InstallExecuteSequence", |
| 407 | [("InitialTargetDir", 'TARGETDIR=""', 750), |
| 408 | ("SetDLLDirToSystem32", 'DLLDIR="" and ' + sys32cond, 751), |
| 409 | ("SetDLLDirToTarget", 'DLLDIR="" and not ' + sys32cond, 752), |
Martin v. Löwis | dff68d0 | 2004-09-10 09:20:10 +0000 | [diff] [blame] | 410 | ("UpdateEditIDLE", None, 1050), |
Martin v. Löwis | 8ffe9ab | 2004-08-22 13:34:34 +0000 | [diff] [blame] | 411 | ("CompilePyc", "COMPILEALL", 6800), |
| 412 | ("CompilePyo", "COMPILEALL", 6801), |
| 413 | ]) |
| 414 | add_data(db, "AdminExecuteSequence", |
| 415 | [("InitialTargetDir", 'TARGETDIR=""', 750), |
| 416 | ("SetDLLDirToTarget", 'DLLDIR=""', 751), |
| 417 | ("CompilePyc", "COMPILEALL", 6800), |
| 418 | ("CompilePyo", "COMPILEALL", 6801), |
| 419 | ]) |
| 420 | |
| 421 | ##################################################################### |
| 422 | # Standard dialogs: FatalError, UserExit, ExitDialog |
| 423 | fatal=PyDialog(db, "FatalError", x, y, w, h, modal, title, |
| 424 | "Finish", "Finish", "Finish") |
| 425 | fatal.title("[ProductName] Installer ended prematurely") |
| 426 | fatal.back("< Back", "Finish", active = 0) |
| 427 | fatal.cancel("Cancel", "Back", active = 0) |
| 428 | fatal.text("Description1", 135, 70, 220, 80, 0x30003, |
| 429 | "[ProductName] setup ended prematurely because of an error. Your system has not been modified. To install this program at a later time, please run the installation again.") |
| 430 | fatal.text("Description2", 135, 155, 220, 20, 0x30003, |
| 431 | "Click the Finish button to exit the Installer.") |
| 432 | c=fatal.next("Finish", "Cancel", name="Finish") |
| 433 | # See "ControlEvent Table". Parameters are the event, the parameter |
| 434 | # to the action, and optionally the condition for the event, and the order |
| 435 | # of events. |
| 436 | c.event("EndDialog", "Exit") |
Tim Peters | 66cb018 | 2004-08-26 05:23:19 +0000 | [diff] [blame] | 437 | |
Martin v. Löwis | 8ffe9ab | 2004-08-22 13:34:34 +0000 | [diff] [blame] | 438 | user_exit=PyDialog(db, "UserExit", x, y, w, h, modal, title, |
| 439 | "Finish", "Finish", "Finish") |
| 440 | user_exit.title("[ProductName] Installer was interrupted") |
| 441 | user_exit.back("< Back", "Finish", active = 0) |
| 442 | user_exit.cancel("Cancel", "Back", active = 0) |
| 443 | user_exit.text("Description1", 135, 70, 220, 80, 0x30003, |
| 444 | "[ProductName] setup was interrupted. Your system has not been modified. " |
| 445 | "To install this program at a later time, please run the installation again.") |
| 446 | user_exit.text("Description2", 135, 155, 220, 20, 0x30003, |
| 447 | "Click the Finish button to exit the Installer.") |
| 448 | c = user_exit.next("Finish", "Cancel", name="Finish") |
| 449 | c.event("EndDialog", "Exit") |
Tim Peters | 66cb018 | 2004-08-26 05:23:19 +0000 | [diff] [blame] | 450 | |
Martin v. Löwis | 8ffe9ab | 2004-08-22 13:34:34 +0000 | [diff] [blame] | 451 | exit_dialog = PyDialog(db, "ExitDialog", x, y, w, h, modal, title, |
| 452 | "Finish", "Finish", "Finish") |
| 453 | exit_dialog.title("Completing the [ProductName] Installer") |
| 454 | exit_dialog.back("< Back", "Finish", active = 0) |
| 455 | exit_dialog.cancel("Cancel", "Back", active = 0) |
Martin v. Löwis | 2dd2a28 | 2004-08-22 17:10:12 +0000 | [diff] [blame] | 456 | exit_dialog.text("Acknowledgements", 135, 95, 220, 120, 0x30003, |
| 457 | "Special Windows thanks to:\n" |
Martin v. Löwis | d3f61a2 | 2004-08-30 09:22:30 +0000 | [diff] [blame] | 458 | " Mark Hammond, without whose years of freely \n" |
| 459 | " shared Windows expertise, Python for Windows \n" |
| 460 | " would still be Python for DOS.") |
Tim Peters | 66cb018 | 2004-08-26 05:23:19 +0000 | [diff] [blame] | 461 | |
Martin v. Löwis | 8c7c56e | 2006-03-05 14:04:26 +0000 | [diff] [blame] | 462 | c = exit_dialog.text("warning", 135, 200, 220, 40, 0x30003, |
| 463 | "{\\VerdanaRed9}Warning: Python 2.5.x is the last " |
| 464 | "Python release for Windows 9x.") |
Martin v. Löwis | df51179 | 2006-03-28 07:51:51 +0000 | [diff] [blame] | 465 | c.condition("Hide", "NOT Version9X") |
Martin v. Löwis | 8c7c56e | 2006-03-05 14:04:26 +0000 | [diff] [blame] | 466 | |
Martin v. Löwis | 2dd2a28 | 2004-08-22 17:10:12 +0000 | [diff] [blame] | 467 | exit_dialog.text("Description", 135, 235, 220, 20, 0x30003, |
Martin v. Löwis | 8ffe9ab | 2004-08-22 13:34:34 +0000 | [diff] [blame] | 468 | "Click the Finish button to exit the Installer.") |
| 469 | c = exit_dialog.next("Finish", "Cancel", name="Finish") |
| 470 | c.event("EndDialog", "Return") |
| 471 | |
| 472 | ##################################################################### |
| 473 | # Required dialog: FilesInUse, ErrorDlg |
| 474 | inuse = PyDialog(db, "FilesInUse", |
| 475 | x, y, w, h, |
| 476 | 19, # KeepModeless|Modal|Visible |
| 477 | title, |
| 478 | "Retry", "Retry", "Retry", bitmap=False) |
| 479 | inuse.text("Title", 15, 6, 200, 15, 0x30003, |
| 480 | r"{\DlgFontBold8}Files in Use") |
| 481 | inuse.text("Description", 20, 23, 280, 20, 0x30003, |
| 482 | "Some files that need to be updated are currently in use.") |
| 483 | inuse.text("Text", 20, 55, 330, 50, 3, |
| 484 | "The following applications are using files that need to be updated by this setup. Close these applications and then click Retry to continue the installation or Cancel to exit it.") |
| 485 | inuse.control("List", "ListBox", 20, 107, 330, 130, 7, "FileInUseProcess", |
| 486 | None, None, None) |
| 487 | c=inuse.back("Exit", "Ignore", name="Exit") |
| 488 | c.event("EndDialog", "Exit") |
| 489 | c=inuse.next("Ignore", "Retry", name="Ignore") |
| 490 | c.event("EndDialog", "Ignore") |
| 491 | c=inuse.cancel("Retry", "Exit", name="Retry") |
| 492 | c.event("EndDialog","Retry") |
| 493 | |
| 494 | |
| 495 | # See "Error Dialog". See "ICE20" for the required names of the controls. |
| 496 | error = Dialog(db, "ErrorDlg", |
| 497 | 50, 10, 330, 101, |
| 498 | 65543, # Error|Minimize|Modal|Visible |
| 499 | title, |
| 500 | "ErrorText", None, None) |
| 501 | error.text("ErrorText", 50,9,280,48,3, "") |
| 502 | error.control("ErrorIcon", "Icon", 15, 9, 24, 24, 5242881, None, "py.ico", None, None) |
| 503 | error.pushbutton("N",120,72,81,21,3,"No",None).event("EndDialog","ErrorNo") |
| 504 | error.pushbutton("Y",240,72,81,21,3,"Yes",None).event("EndDialog","ErrorYes") |
| 505 | error.pushbutton("A",0,72,81,21,3,"Abort",None).event("EndDialog","ErrorAbort") |
| 506 | error.pushbutton("C",42,72,81,21,3,"Cancel",None).event("EndDialog","ErrorCancel") |
| 507 | error.pushbutton("I",81,72,81,21,3,"Ignore",None).event("EndDialog","ErrorIgnore") |
| 508 | error.pushbutton("O",159,72,81,21,3,"Ok",None).event("EndDialog","ErrorOk") |
| 509 | error.pushbutton("R",198,72,81,21,3,"Retry",None).event("EndDialog","ErrorRetry") |
| 510 | |
| 511 | ##################################################################### |
| 512 | # Global "Query Cancel" dialog |
| 513 | cancel = Dialog(db, "CancelDlg", 50, 10, 260, 85, 3, title, |
| 514 | "No", "No", "No") |
Tim Peters | 66cb018 | 2004-08-26 05:23:19 +0000 | [diff] [blame] | 515 | cancel.text("Text", 48, 15, 194, 30, 3, |
Martin v. Löwis | 8ffe9ab | 2004-08-22 13:34:34 +0000 | [diff] [blame] | 516 | "Are you sure you want to cancel [ProductName] installation?") |
| 517 | cancel.control("Icon", "Icon", 15, 15, 24, 24, 5242881, None, |
| 518 | "py.ico", None, None) |
| 519 | c=cancel.pushbutton("Yes", 72, 57, 56, 17, 3, "Yes", "No") |
| 520 | c.event("EndDialog", "Exit") |
Tim Peters | 66cb018 | 2004-08-26 05:23:19 +0000 | [diff] [blame] | 521 | |
Martin v. Löwis | 8ffe9ab | 2004-08-22 13:34:34 +0000 | [diff] [blame] | 522 | c=cancel.pushbutton("No", 132, 57, 56, 17, 3, "No", "Yes") |
| 523 | c.event("EndDialog", "Return") |
| 524 | |
| 525 | ##################################################################### |
| 526 | # Global "Wait for costing" dialog |
| 527 | costing = Dialog(db, "WaitForCostingDlg", 50, 10, 260, 85, modal, title, |
| 528 | "Return", "Return", "Return") |
| 529 | costing.text("Text", 48, 15, 194, 30, 3, |
| 530 | "Please wait while the installer finishes determining your disk space requirements.") |
| 531 | costing.control("Icon", "Icon", 15, 15, 24, 24, 5242881, None, |
| 532 | "py.ico", None, None) |
| 533 | c = costing.pushbutton("Return", 102, 57, 56, 17, 3, "Return", None) |
| 534 | c.event("EndDialog", "Exit") |
| 535 | |
| 536 | ##################################################################### |
| 537 | # Preparation dialog: no user input except cancellation |
| 538 | prep = PyDialog(db, "PrepareDlg", x, y, w, h, modeless, title, |
| 539 | "Cancel", "Cancel", "Cancel") |
| 540 | prep.text("Description", 135, 70, 220, 40, 0x30003, |
| 541 | "Please wait while the Installer prepares to guide you through the installation.") |
| 542 | prep.title("Welcome to the [ProductName] Installer") |
| 543 | c=prep.text("ActionText", 135, 110, 220, 20, 0x30003, "Pondering...") |
| 544 | c.mapping("ActionText", "Text") |
| 545 | c=prep.text("ActionData", 135, 135, 220, 30, 0x30003, None) |
| 546 | c.mapping("ActionData", "Text") |
| 547 | prep.back("Back", None, active=0) |
| 548 | prep.next("Next", None, active=0) |
| 549 | c=prep.cancel("Cancel", None) |
| 550 | c.event("SpawnDialog", "CancelDlg") |
| 551 | |
| 552 | ##################################################################### |
| 553 | # Target directory selection |
| 554 | seldlg = PyDialog(db, "SelectDirectoryDlg", x, y, w, h, modal, title, |
| 555 | "Next", "Next", "Cancel") |
| 556 | seldlg.title("Select Destination Directory") |
Martin v. Löwis | 141f41a | 2005-03-15 00:39:40 +0000 | [diff] [blame] | 557 | c = seldlg.text("Existing", 135, 25, 235, 30, 0x30003, |
| 558 | "{\VerdanaRed9}This update will replace your existing [ProductLine] installation.") |
| 559 | c.condition("Hide", 'REMOVEOLDVERSION="" and REMOVEOLDSNAPSHOT=""') |
Martin v. Löwis | 8ffe9ab | 2004-08-22 13:34:34 +0000 | [diff] [blame] | 560 | seldlg.text("Description", 135, 50, 220, 40, 0x30003, |
| 561 | "Please select a directory for the [ProductName] files.") |
| 562 | |
| 563 | seldlg.back("< Back", None, active=0) |
| 564 | c = seldlg.next("Next >", "Cancel") |
| 565 | c.event("DoAction", "CheckDir", "TargetExistsOk<>1", order=1) |
| 566 | # If the target exists, but we found that we are going to remove old versions, don't bother |
| 567 | # confirming that the target directory exists. Strictly speaking, we should determine that |
| 568 | # the target directory is indeed the target of the product that we are going to remove, but |
| 569 | # I don't know how to do that. |
| 570 | c.event("SpawnDialog", "ExistingDirectoryDlg", 'TargetExists=1 and REMOVEOLDVERSION="" and REMOVEOLDSNAPSHOT=""', 2) |
| 571 | c.event("SetTargetPath", "TARGETDIR", 'TargetExists=0 or REMOVEOLDVERSION<>"" or REMOVEOLDSNAPSHOT<>""', 3) |
| 572 | c.event("SpawnWaitDialog", "WaitForCostingDlg", "CostingComplete=1", 4) |
| 573 | c.event("NewDialog", "SelectFeaturesDlg", 'TargetExists=0 or REMOVEOLDVERSION<>"" or REMOVEOLDSNAPSHOT<>""', 5) |
| 574 | |
| 575 | c = seldlg.cancel("Cancel", "DirectoryCombo") |
| 576 | c.event("SpawnDialog", "CancelDlg") |
| 577 | |
| 578 | seldlg.control("DirectoryCombo", "DirectoryCombo", 135, 70, 172, 80, 393219, |
| 579 | "TARGETDIR", None, "DirectoryList", None) |
| 580 | seldlg.control("DirectoryList", "DirectoryList", 135, 90, 208, 136, 3, "TARGETDIR", |
| 581 | None, "PathEdit", None) |
| 582 | seldlg.control("PathEdit", "PathEdit", 135, 230, 206, 16, 3, "TARGETDIR", None, "Next", None) |
| 583 | c = seldlg.pushbutton("Up", 306, 70, 18, 18, 3, "Up", None) |
| 584 | c.event("DirectoryListUp", "0") |
| 585 | c = seldlg.pushbutton("NewDir", 324, 70, 30, 18, 3, "New", None) |
| 586 | c.event("DirectoryListNew", "0") |
| 587 | |
| 588 | ##################################################################### |
| 589 | # SelectFeaturesDlg |
| 590 | features = PyDialog(db, "SelectFeaturesDlg", x, y, w, h, modal|track_disk_space, |
| 591 | title, "Tree", "Next", "Cancel") |
| 592 | features.title("Customize [ProductName]") |
| 593 | features.text("Description", 135, 35, 220, 15, 0x30003, |
| 594 | "Select the way you want features to be installed.") |
| 595 | features.text("Text", 135,45,220,30, 3, |
| 596 | "Click on the icons in the tree below to change the way features will be installed.") |
| 597 | |
| 598 | c=features.back("< Back", "Next") |
| 599 | c.event("NewDialog", "SelectDirectoryDlg") |
| 600 | |
| 601 | c=features.next("Next >", "Cancel") |
| 602 | c.mapping("SelectionNoItems", "Enabled") |
| 603 | c.event("SpawnDialog", "DiskCostDlg", "OutOfDiskSpace=1", order=1) |
| 604 | c.event("EndDialog", "Return", "OutOfDiskSpace<>1", order=2) |
| 605 | |
| 606 | c=features.cancel("Cancel", "Tree") |
| 607 | c.event("SpawnDialog", "CancelDlg") |
| 608 | |
Tim Peters | 66cb018 | 2004-08-26 05:23:19 +0000 | [diff] [blame] | 609 | # The browse property is not used, since we have only a single target path (selected already) |
Martin v. Löwis | 8ffe9ab | 2004-08-22 13:34:34 +0000 | [diff] [blame] | 610 | features.control("Tree", "SelectionTree", 135, 75, 220, 95, 7, "_BrowseProperty", |
| 611 | "Tree of selections", "Back", None) |
| 612 | |
| 613 | #c=features.pushbutton("Reset", 42, 243, 56, 17, 3, "Reset", "DiskCost") |
| 614 | #c.mapping("SelectionNoItems", "Enabled") |
| 615 | #c.event("Reset", "0") |
Tim Peters | 66cb018 | 2004-08-26 05:23:19 +0000 | [diff] [blame] | 616 | |
Martin v. Löwis | 8ffe9ab | 2004-08-22 13:34:34 +0000 | [diff] [blame] | 617 | features.control("Box", "GroupBox", 135, 170, 225, 90, 1, None, None, None, None) |
| 618 | |
| 619 | c=features.xbutton("DiskCost", "Disk &Usage", None, 0.10) |
| 620 | c.mapping("SelectionNoItems","Enabled") |
| 621 | c.event("SpawnDialog", "DiskCostDlg") |
| 622 | |
| 623 | c=features.xbutton("Advanced", "Advanced", None, 0.30) |
| 624 | c.event("SpawnDialog", "AdvancedDlg") |
| 625 | |
| 626 | c=features.text("ItemDescription", 140, 180, 210, 30, 3, |
| 627 | "Multiline description of the currently selected item.") |
| 628 | c.mapping("SelectionDescription","Text") |
Tim Peters | 66cb018 | 2004-08-26 05:23:19 +0000 | [diff] [blame] | 629 | |
Martin v. Löwis | 8ffe9ab | 2004-08-22 13:34:34 +0000 | [diff] [blame] | 630 | c=features.text("ItemSize", 140, 210, 210, 45, 3, |
| 631 | "The size of the currently selected item.") |
| 632 | c.mapping("SelectionSize", "Text") |
| 633 | |
| 634 | ##################################################################### |
| 635 | # Disk cost |
| 636 | cost = PyDialog(db, "DiskCostDlg", x, y, w, h, modal, title, |
| 637 | "OK", "OK", "OK", bitmap=False) |
| 638 | cost.text("Title", 15, 6, 200, 15, 0x30003, |
| 639 | "{\DlgFontBold8}Disk Space Requirements") |
| 640 | cost.text("Description", 20, 20, 280, 20, 0x30003, |
| 641 | "The disk space required for the installation of the selected features.") |
| 642 | cost.text("Text", 20, 53, 330, 60, 3, |
| 643 | "The highlighted volumes (if any) do not have enough disk space " |
| 644 | "available for the currently selected features. You can either " |
| 645 | "remove some files from the highlighted volumes, or choose to " |
| 646 | "install less features onto local drive(s), or select different " |
| 647 | "destination drive(s).") |
| 648 | cost.control("VolumeList", "VolumeCostList", 20, 100, 330, 150, 393223, |
| 649 | None, "{120}{70}{70}{70}{70}", None, None) |
| 650 | cost.xbutton("OK", "Ok", None, 0.5).event("EndDialog", "Return") |
| 651 | |
| 652 | ##################################################################### |
| 653 | # WhichUsers Dialog. Only available on NT, and for privileged users. |
| 654 | # This must be run before FindRelatedProducts, because that will |
| 655 | # take into account whether the previous installation was per-user |
| 656 | # or per-machine. We currently don't support going back to this |
| 657 | # dialog after "Next" was selected; to support this, we would need to |
| 658 | # find how to reset the ALLUSERS property, and how to re-run |
| 659 | # FindRelatedProducts. |
| 660 | # On Windows9x, the ALLUSERS property is ignored on the command line |
| 661 | # and in the Property table, but installer fails according to the documentation |
| 662 | # if a dialog attempts to set ALLUSERS. |
| 663 | whichusers = PyDialog(db, "WhichUsersDlg", x, y, w, h, modal, title, |
| 664 | "AdminInstall", "Next", "Cancel") |
| 665 | whichusers.title("Select whether to install [ProductName] for all users of this computer.") |
| 666 | # A radio group with two options: allusers, justme |
| 667 | g = whichusers.radiogroup("AdminInstall", 135, 60, 160, 50, 3, |
| 668 | "WhichUsers", "", "Next") |
| 669 | g.add("ALL", 0, 5, 150, 20, "Install for all users") |
| 670 | g.add("JUSTME", 0, 25, 150, 20, "Install just for me") |
| 671 | |
Tim Peters | 66cb018 | 2004-08-26 05:23:19 +0000 | [diff] [blame] | 672 | whichusers.back("Back", None, active=0) |
Martin v. Löwis | 8ffe9ab | 2004-08-22 13:34:34 +0000 | [diff] [blame] | 673 | |
| 674 | c = whichusers.next("Next >", "Cancel") |
| 675 | c.event("[ALLUSERS]", "1", 'WhichUsers="ALL"', 1) |
| 676 | c.event("EndDialog", "Return", order = 2) |
| 677 | |
| 678 | c = whichusers.cancel("Cancel", "AdminInstall") |
| 679 | c.event("SpawnDialog", "CancelDlg") |
Tim Peters | 66cb018 | 2004-08-26 05:23:19 +0000 | [diff] [blame] | 680 | |
Martin v. Löwis | 8ffe9ab | 2004-08-22 13:34:34 +0000 | [diff] [blame] | 681 | ##################################################################### |
| 682 | # Advanced Dialog. |
| 683 | advanced = PyDialog(db, "AdvancedDlg", x, y, w, h, modal, title, |
| 684 | "CompilePyc", "Next", "Cancel") |
| 685 | advanced.title("Advanced Options for [ProductName]") |
| 686 | # A radio group with two options: allusers, justme |
| 687 | advanced.checkbox("CompilePyc", 135, 60, 230, 50, 3, |
| 688 | "COMPILEALL", "Compile .py files to byte code after installation", "Next") |
| 689 | |
| 690 | c = advanced.next("Finish", "Cancel") |
| 691 | c.event("EndDialog", "Return") |
| 692 | |
| 693 | c = advanced.cancel("Cancel", "CompilePyc") |
| 694 | c.event("SpawnDialog", "CancelDlg") |
Tim Peters | 66cb018 | 2004-08-26 05:23:19 +0000 | [diff] [blame] | 695 | |
Martin v. Löwis | 8ffe9ab | 2004-08-22 13:34:34 +0000 | [diff] [blame] | 696 | ##################################################################### |
Tim Peters | 66cb018 | 2004-08-26 05:23:19 +0000 | [diff] [blame] | 697 | # Existing Directory dialog |
Martin v. Löwis | 8ffe9ab | 2004-08-22 13:34:34 +0000 | [diff] [blame] | 698 | dlg = Dialog(db, "ExistingDirectoryDlg", 50, 30, 200, 80, modal, title, |
| 699 | "No", "No", "No") |
| 700 | dlg.text("Title", 10, 20, 180, 40, 3, |
| 701 | "[TARGETDIR] exists. Are you sure you want to overwrite existing files?") |
| 702 | c=dlg.pushbutton("Yes", 30, 60, 55, 17, 3, "Yes", "No") |
| 703 | c.event("[TargetExists]", "0", order=1) |
| 704 | c.event("[TargetExistsOk]", "1", order=2) |
| 705 | c.event("EndDialog", "Return", order=3) |
| 706 | c=dlg.pushbutton("No", 115, 60, 55, 17, 3, "No", "Yes") |
| 707 | c.event("EndDialog", "Return") |
| 708 | |
| 709 | ##################################################################### |
| 710 | # Installation Progress dialog (modeless) |
| 711 | progress = PyDialog(db, "ProgressDlg", x, y, w, h, modeless, title, |
| 712 | "Cancel", "Cancel", "Cancel", bitmap=False) |
| 713 | progress.text("Title", 20, 15, 200, 15, 0x30003, |
| 714 | "{\DlgFontBold8}[Progress1] [ProductName]") |
| 715 | progress.text("Text", 35, 65, 300, 30, 3, |
| 716 | "Please wait while the Installer [Progress2] [ProductName]. " |
| 717 | "This may take several minutes.") |
| 718 | progress.text("StatusLabel", 35, 100, 35, 20, 3, "Status:") |
| 719 | |
| 720 | c=progress.text("ActionText", 70, 100, w-70, 20, 3, "Pondering...") |
| 721 | c.mapping("ActionText", "Text") |
| 722 | |
| 723 | #c=progress.text("ActionData", 35, 140, 300, 20, 3, None) |
| 724 | #c.mapping("ActionData", "Text") |
| 725 | |
| 726 | c=progress.control("ProgressBar", "ProgressBar", 35, 120, 300, 10, 65537, |
| 727 | None, "Progress done", None, None) |
| 728 | c.mapping("SetProgress", "Progress") |
| 729 | |
| 730 | progress.back("< Back", "Next", active=False) |
| 731 | progress.next("Next >", "Cancel", active=False) |
| 732 | progress.cancel("Cancel", "Back").event("SpawnDialog", "CancelDlg") |
| 733 | |
| 734 | # Maintenance type: repair/uninstall |
| 735 | maint = PyDialog(db, "MaintenanceTypeDlg", x, y, w, h, modal, title, |
| 736 | "Next", "Next", "Cancel") |
| 737 | maint.title("Welcome to the [ProductName] Setup Wizard") |
| 738 | maint.text("BodyText", 135, 63, 230, 42, 3, |
| 739 | "Select whether you want to repair or remove [ProductName].") |
| 740 | g=maint.radiogroup("RepairRadioGroup", 135, 108, 230, 60, 3, |
| 741 | "MaintenanceForm_Action", "", "Next") |
| 742 | g.add("Change", 0, 0, 200, 17, "&Change [ProductName]") |
| 743 | g.add("Repair", 0, 18, 200, 17, "&Repair [ProductName]") |
| 744 | g.add("Remove", 0, 36, 200, 17, "Re&move [ProductName]") |
Tim Peters | 66cb018 | 2004-08-26 05:23:19 +0000 | [diff] [blame] | 745 | |
Martin v. Löwis | 8ffe9ab | 2004-08-22 13:34:34 +0000 | [diff] [blame] | 746 | maint.back("< Back", None, active=False) |
| 747 | c=maint.next("Finish", "Cancel") |
| 748 | # Change installation: Change progress dialog to "Change", then ask |
| 749 | # for feature selection |
| 750 | c.event("[Progress1]", "Change", 'MaintenanceForm_Action="Change"', 1) |
| 751 | c.event("[Progress2]", "changes", 'MaintenanceForm_Action="Change"', 2) |
| 752 | |
| 753 | # Reinstall: Change progress dialog to "Repair", then invoke reinstall |
| 754 | # Also set list of reinstalled features to "ALL" |
| 755 | c.event("[REINSTALL]", "ALL", 'MaintenanceForm_Action="Repair"', 5) |
| 756 | c.event("[Progress1]", "Repairing", 'MaintenanceForm_Action="Repair"', 6) |
Raymond Hettinger | 72f0801 | 2004-11-07 07:08:25 +0000 | [diff] [blame] | 757 | c.event("[Progress2]", "repairs", 'MaintenanceForm_Action="Repair"', 7) |
Martin v. Löwis | 8ffe9ab | 2004-08-22 13:34:34 +0000 | [diff] [blame] | 758 | c.event("Reinstall", "ALL", 'MaintenanceForm_Action="Repair"', 8) |
| 759 | |
| 760 | # Uninstall: Change progress to "Remove", then invoke uninstall |
| 761 | # Also set list of removed features to "ALL" |
| 762 | c.event("[REMOVE]", "ALL", 'MaintenanceForm_Action="Remove"', 11) |
| 763 | c.event("[Progress1]", "Removing", 'MaintenanceForm_Action="Remove"', 12) |
| 764 | c.event("[Progress2]", "removes", 'MaintenanceForm_Action="Remove"', 13) |
| 765 | c.event("Remove", "ALL", 'MaintenanceForm_Action="Remove"', 14) |
| 766 | |
Tim Peters | 66cb018 | 2004-08-26 05:23:19 +0000 | [diff] [blame] | 767 | # Close dialog when maintenance action scheduled |
Martin v. Löwis | 8ffe9ab | 2004-08-22 13:34:34 +0000 | [diff] [blame] | 768 | c.event("EndDialog", "Return", 'MaintenanceForm_Action<>"Change"', 20) |
| 769 | c.event("NewDialog", "SelectFeaturesDlg", 'MaintenanceForm_Action="Change"', 21) |
Tim Peters | 66cb018 | 2004-08-26 05:23:19 +0000 | [diff] [blame] | 770 | |
Martin v. Löwis | 8ffe9ab | 2004-08-22 13:34:34 +0000 | [diff] [blame] | 771 | maint.cancel("Cancel", "RepairRadioGroup").event("SpawnDialog", "CancelDlg") |
Tim Peters | 66cb018 | 2004-08-26 05:23:19 +0000 | [diff] [blame] | 772 | |
Martin v. Löwis | 8ffe9ab | 2004-08-22 13:34:34 +0000 | [diff] [blame] | 773 | |
| 774 | # See "Feature Table". The feature level is 1 for all features, |
| 775 | # and the feature attributes are 0 for the DefaultFeature, and |
| 776 | # FollowParent for all other features. The numbers are the Display |
| 777 | # column. |
| 778 | def add_features(db): |
| 779 | # feature attributes: |
| 780 | # msidbFeatureAttributesFollowParent == 2 |
| 781 | # msidbFeatureAttributesDisallowAdvertise == 8 |
| 782 | # Features that need to be installed with together with the main feature |
| 783 | # (i.e. additional Python libraries) need to follow the parent feature. |
| 784 | # Features that have no advertisement trigger (e.g. the test suite) |
| 785 | # must not support advertisement |
| 786 | global default_feature, tcltk, htmlfiles, tools, testsuite, ext_feature |
| 787 | default_feature = Feature(db, "DefaultFeature", "Python", |
| 788 | "Python Interpreter and Libraries", |
| 789 | 1, directory = "TARGETDIR") |
Martin v. Löwis | dff68d0 | 2004-09-10 09:20:10 +0000 | [diff] [blame] | 790 | # We don't support advertisement of extensions |
Martin v. Löwis | 8ffe9ab | 2004-08-22 13:34:34 +0000 | [diff] [blame] | 791 | ext_feature = Feature(db, "Extensions", "Register Extensions", |
| 792 | "Make this Python installation the default Python installation", 3, |
Martin v. Löwis | dff68d0 | 2004-09-10 09:20:10 +0000 | [diff] [blame] | 793 | parent = default_feature, attributes=2|8) |
Martin v. Löwis | e0f780d | 2004-09-01 14:51:06 +0000 | [diff] [blame] | 794 | if have_tcl: |
| 795 | tcltk = Feature(db, "TclTk", "Tcl/Tk", "Tkinter, IDLE, pydoc", 5, |
Martin v. Löwis | 8ffe9ab | 2004-08-22 13:34:34 +0000 | [diff] [blame] | 796 | parent = default_feature, attributes=2) |
| 797 | htmlfiles = Feature(db, "Documentation", "Documentation", |
| 798 | "Python HTMLHelp File", 7, parent = default_feature) |
| 799 | tools = Feature(db, "Tools", "Utility Scripts", |
Tim Peters | 66cb018 | 2004-08-26 05:23:19 +0000 | [diff] [blame] | 800 | "Python utility scripts (Tools/", 9, |
Martin v. Löwis | 8ffe9ab | 2004-08-22 13:34:34 +0000 | [diff] [blame] | 801 | parent = default_feature, attributes=2) |
| 802 | testsuite = Feature(db, "Testsuite", "Test suite", |
| 803 | "Python test suite (Lib/test/)", 11, |
| 804 | parent = default_feature, attributes=2|8) |
Tim Peters | 66cb018 | 2004-08-26 05:23:19 +0000 | [diff] [blame] | 805 | |
Martin v. Löwis | 8ffe9ab | 2004-08-22 13:34:34 +0000 | [diff] [blame] | 806 | def extract_msvcr71(): |
| 807 | import _winreg |
| 808 | # Find the location of the merge modules |
| 809 | k = _winreg.OpenKey( |
| 810 | _winreg.HKEY_LOCAL_MACHINE, |
| 811 | r"Software\Microsoft\VisualStudio\7.1\Setup\VS") |
| 812 | dir = _winreg.QueryValueEx(k, "MSMDir")[0] |
| 813 | _winreg.CloseKey(k) |
| 814 | files = glob.glob1(dir, "*CRT71*") |
| 815 | assert len(files) == 1 |
| 816 | file = os.path.join(dir, files[0]) |
| 817 | # Extract msvcr71.dll |
| 818 | m = msilib.MakeMerge2() |
| 819 | m.OpenModule(file, 0) |
| 820 | m.ExtractFiles(".") |
| 821 | m.CloseModule() |
| 822 | # Find the version/language of msvcr71.dll |
| 823 | installer = msilib.MakeInstaller() |
| 824 | return installer.FileVersion("msvcr71.dll", 0), \ |
| 825 | installer.FileVersion("msvcr71.dll", 1) |
| 826 | |
| 827 | class PyDirectory(Directory): |
| 828 | """By default, all components in the Python installer |
| 829 | can run from source.""" |
| 830 | def __init__(self, *args, **kw): |
| 831 | if not kw.has_key("componentflags"): |
| 832 | kw['componentflags'] = 2 #msidbComponentAttributesOptional |
| 833 | Directory.__init__(self, *args, **kw) |
| 834 | |
| 835 | # See "File Table", "Component Table", "Directory Table", |
| 836 | # "FeatureComponents Table" |
| 837 | def add_files(db): |
| 838 | cab = CAB("python") |
| 839 | tmpfiles = [] |
| 840 | # Add all executables, icons, text files into the TARGETDIR component |
| 841 | root = PyDirectory(db, cab, None, srcdir, "TARGETDIR", "SourceDir") |
| 842 | default_feature.set_current() |
Martin v. Löwis | e0f780d | 2004-09-01 14:51:06 +0000 | [diff] [blame] | 843 | if not msilib.Win64: |
| 844 | root.add_file("PCBuild/w9xpopen.exe") |
Martin v. Löwis | 8ffe9ab | 2004-08-22 13:34:34 +0000 | [diff] [blame] | 845 | root.add_file("README.txt", src="README") |
| 846 | root.add_file("NEWS.txt", src="Misc/NEWS") |
| 847 | root.add_file("LICENSE.txt", src="LICENSE") |
| 848 | root.start_component("python.exe", keyfile="python.exe") |
| 849 | root.add_file("PCBuild/python.exe") |
| 850 | root.start_component("pythonw.exe", keyfile="pythonw.exe") |
| 851 | root.add_file("PCBuild/pythonw.exe") |
| 852 | |
| 853 | # msidbComponentAttributesSharedDllRefCount = 8, see "Component Table" |
| 854 | dlldir = PyDirectory(db, cab, root, srcdir, "DLLDIR", ".") |
| 855 | pydll = "python%s%s.dll" % (major, minor) |
| 856 | pydllsrc = srcdir + "/PCBuild/" + pydll |
Martin v. Löwis | 141f41a | 2005-03-15 00:39:40 +0000 | [diff] [blame] | 857 | dlldir.start_component("DLLDIR", flags = 8, keyfile = pydll, uuid = pythondll_uuid) |
Martin v. Löwis | 8ffe9ab | 2004-08-22 13:34:34 +0000 | [diff] [blame] | 858 | installer = msilib.MakeInstaller() |
| 859 | pyversion = installer.FileVersion(pydllsrc, 0) |
| 860 | if not snapshot: |
| 861 | # For releases, the Python DLL has the same version as the |
| 862 | # installer package. |
| 863 | assert pyversion.split(".")[:3] == current_version.split(".") |
| 864 | dlldir.add_file("PCBuild/python%s%s.dll" % (major, minor), |
| 865 | version=pyversion, |
| 866 | language=installer.FileVersion(pydllsrc, 1)) |
| 867 | # XXX determine dependencies |
| 868 | version, lang = extract_msvcr71() |
Martin v. Löwis | 141f41a | 2005-03-15 00:39:40 +0000 | [diff] [blame] | 869 | dlldir.start_component("msvcr71", flags=8, keyfile="msvcr71.dll", uuid=msvcr71_uuid) |
Martin v. Löwis | 8ffe9ab | 2004-08-22 13:34:34 +0000 | [diff] [blame] | 870 | dlldir.add_file("msvcr71.dll", src=os.path.abspath("msvcr71.dll"), |
| 871 | version=version, language=lang) |
| 872 | tmpfiles.append("msvcr71.dll") |
Tim Peters | 66cb018 | 2004-08-26 05:23:19 +0000 | [diff] [blame] | 873 | |
Martin v. Löwis | 8ffe9ab | 2004-08-22 13:34:34 +0000 | [diff] [blame] | 874 | # Add all .py files in Lib, except lib-tk, test |
| 875 | dirs={} |
| 876 | pydirs = [(root,"Lib")] |
| 877 | while pydirs: |
| 878 | parent, dir = pydirs.pop() |
Martin v. Löwis | 9ca9f56 | 2006-01-03 06:29:53 +0000 | [diff] [blame] | 879 | if dir == ".svn" or dir.startswith("plat-"): |
Martin v. Löwis | 8ffe9ab | 2004-08-22 13:34:34 +0000 | [diff] [blame] | 880 | continue |
| 881 | elif dir in ["lib-tk", "idlelib", "Icons"]: |
Martin v. Löwis | e0f780d | 2004-09-01 14:51:06 +0000 | [diff] [blame] | 882 | if not have_tcl: |
| 883 | continue |
Martin v. Löwis | 8ffe9ab | 2004-08-22 13:34:34 +0000 | [diff] [blame] | 884 | tcltk.set_current() |
Martin v. Löwis | 5c9e55e | 2004-12-30 14:08:18 +0000 | [diff] [blame] | 885 | elif dir in ['test', 'tests', 'data', 'output']: |
Martin v. Löwis | 59c3acc | 2006-04-03 12:07:46 +0000 | [diff] [blame] | 886 | # test: Lib, Lib/email, Lib/bsddb, Lib/ctypes, Lib/sqlite3 |
Martin v. Löwis | 5c9e55e | 2004-12-30 14:08:18 +0000 | [diff] [blame] | 887 | # tests: Lib/distutils |
| 888 | # data: Lib/email/test |
| 889 | # output: Lib/test |
Martin v. Löwis | 8ffe9ab | 2004-08-22 13:34:34 +0000 | [diff] [blame] | 890 | testsuite.set_current() |
| 891 | else: |
| 892 | default_feature.set_current() |
| 893 | lib = PyDirectory(db, cab, parent, dir, dir, "%s|%s" % (parent.make_short(dir), dir)) |
| 894 | # Add additional files |
| 895 | dirs[dir]=lib |
| 896 | lib.glob("*.txt") |
| 897 | if dir=='site-packages': |
Martin v. Löwis | 6d60c09 | 2004-11-21 10:16:26 +0000 | [diff] [blame] | 898 | lib.add_file("README.txt", src="README") |
Martin v. Löwis | 8ffe9ab | 2004-08-22 13:34:34 +0000 | [diff] [blame] | 899 | continue |
| 900 | files = lib.glob("*.py") |
| 901 | files += lib.glob("*.pyw") |
| 902 | if files: |
| 903 | # Add an entry to the RemoveFile table to remove bytecode files. |
| 904 | lib.remove_pyc() |
Martin v. Löwis | 64ed043 | 2006-04-21 10:00:46 +0000 | [diff] [blame] | 905 | if dir.endswith('.egg-info'): |
| 906 | lib.add_file('entry_points.txt') |
| 907 | lib.add_file('PKG-INFO') |
| 908 | lib.add_file('top_level.txt') |
| 909 | lib.add_file('zip-safe') |
| 910 | continue |
Martin v. Löwis | 8ffe9ab | 2004-08-22 13:34:34 +0000 | [diff] [blame] | 911 | if dir=='test' and parent.physical=='Lib': |
| 912 | lib.add_file("185test.db") |
| 913 | lib.add_file("audiotest.au") |
| 914 | lib.add_file("cfgparser.1") |
| 915 | lib.add_file("test.xml") |
| 916 | lib.add_file("test.xml.out") |
| 917 | lib.add_file("testtar.tar") |
Martin v. Löwis | 7d3755d | 2004-09-06 06:31:12 +0000 | [diff] [blame] | 918 | lib.add_file("test_difflib_expect.html") |
Martin v. Löwis | 59c3acc | 2006-04-03 12:07:46 +0000 | [diff] [blame] | 919 | lib.add_file("check_soundcard.vbs") |
Thomas Heller | 3bd3315 | 2006-04-04 18:41:13 +0000 | [diff] [blame] | 920 | lib.add_file("empty.vbs") |
Martin v. Löwis | 8ffe9ab | 2004-08-22 13:34:34 +0000 | [diff] [blame] | 921 | lib.glob("*.uue") |
| 922 | lib.add_file("readme.txt", src="README") |
| 923 | if dir=='decimaltestdata': |
| 924 | lib.glob("*.decTest") |
| 925 | if dir=='output': |
| 926 | lib.glob("test_*") |
| 927 | if dir=='idlelib': |
| 928 | lib.glob("*.def") |
| 929 | lib.add_file("idle.bat") |
| 930 | if dir=="Icons": |
| 931 | lib.glob("*.gif") |
| 932 | lib.add_file("idle.icns") |
Martin v. Löwis | 64ed043 | 2006-04-21 10:00:46 +0000 | [diff] [blame] | 933 | if dir=="command" and parent.physical=="distutils": |
Martin v. Löwis | 8ffe9ab | 2004-08-22 13:34:34 +0000 | [diff] [blame] | 934 | lib.add_file("wininst-6.exe") |
| 935 | lib.add_file("wininst-7.1.exe") |
Martin v. Löwis | 64ed043 | 2006-04-21 10:00:46 +0000 | [diff] [blame] | 936 | if dir=="setuptools": |
| 937 | lib.add_file("cli.exe") |
| 938 | lib.add_file("gui.exe") |
Martin v. Löwis | 8ffe9ab | 2004-08-22 13:34:34 +0000 | [diff] [blame] | 939 | if dir=="data" and parent.physical=="test" and parent.basedir.physical=="email": |
Martin v. Löwis | 9ca9f56 | 2006-01-03 06:29:53 +0000 | [diff] [blame] | 940 | # This should contain all non-.svn files listed in subversion |
Martin v. Löwis | 8ffe9ab | 2004-08-22 13:34:34 +0000 | [diff] [blame] | 941 | for f in os.listdir(lib.absolute): |
Martin v. Löwis | 9ca9f56 | 2006-01-03 06:29:53 +0000 | [diff] [blame] | 942 | if f.endswith(".txt") or f==".svn":continue |
Martin v. Löwis | 8ffe9ab | 2004-08-22 13:34:34 +0000 | [diff] [blame] | 943 | if f.endswith(".au") or f.endswith(".gif"): |
| 944 | lib.add_file(f) |
| 945 | else: |
| 946 | print "WARNING: New file %s in email/test/data" % f |
| 947 | for f in os.listdir(lib.absolute): |
| 948 | if os.path.isdir(os.path.join(lib.absolute, f)): |
| 949 | pydirs.append((lib, f)) |
| 950 | # Add DLLs |
| 951 | default_feature.set_current() |
| 952 | lib = PyDirectory(db, cab, root, srcdir+"/PCBuild", "DLLs", "DLLS|DLLs") |
Martin v. Löwis | 1319bb1 | 2006-05-12 13:57:36 +0000 | [diff] [blame] | 953 | lib.add_file("py.ico", src="../PC/py.ico") |
| 954 | lib.add_file("pyc.ico", src="../PC/pyc.ico") |
Martin v. Löwis | 8ffe9ab | 2004-08-22 13:34:34 +0000 | [diff] [blame] | 955 | dlls = [] |
| 956 | tclfiles = [] |
| 957 | for f in extensions: |
| 958 | if f=="_tkinter.pyd": |
| 959 | continue |
| 960 | if not os.path.exists(srcdir+"/PCBuild/"+f): |
| 961 | print "WARNING: Missing extension", f |
| 962 | continue |
| 963 | dlls.append(f) |
| 964 | lib.add_file(f) |
Martin v. Löwis | 88ef637 | 2006-07-06 06:55:58 +0000 | [diff] [blame^] | 965 | # Add sqlite |
| 966 | if msilib.msi_type=="Intel64;1033": |
| 967 | sqlite_arch = "/ia64" |
| 968 | elif msilib.msi_type=="x64;1033": |
| 969 | sqlite_arch = "/amd64" |
| 970 | else: |
| 971 | sqlite_arch = "" |
| 972 | lib.add_file(srcdir+"/"+sqlite_dir+sqlite_arch+"/sqlite3.dll") |
Martin v. Löwis | e0f780d | 2004-09-01 14:51:06 +0000 | [diff] [blame] | 973 | if have_tcl: |
| 974 | if not os.path.exists(srcdir+"/PCBuild/_tkinter.pyd"): |
| 975 | print "WARNING: Missing _tkinter.pyd" |
| 976 | else: |
| 977 | lib.start_component("TkDLLs", tcltk) |
| 978 | lib.add_file("_tkinter.pyd") |
| 979 | dlls.append("_tkinter.pyd") |
| 980 | tcldir = os.path.normpath(srcdir+"/../tcltk/bin") |
| 981 | for f in glob.glob1(tcldir, "*.dll"): |
| 982 | lib.add_file(f, src=os.path.join(tcldir, f)) |
Martin v. Löwis | 8ffe9ab | 2004-08-22 13:34:34 +0000 | [diff] [blame] | 983 | # check whether there are any unknown extensions |
| 984 | for f in glob.glob1(srcdir+"/PCBuild", "*.pyd"): |
| 985 | if f.endswith("_d.pyd"): continue # debug version |
| 986 | if f in dlls: continue |
| 987 | print "WARNING: Unknown extension", f |
Tim Peters | 66cb018 | 2004-08-26 05:23:19 +0000 | [diff] [blame] | 988 | |
Martin v. Löwis | 8ffe9ab | 2004-08-22 13:34:34 +0000 | [diff] [blame] | 989 | # Add headers |
| 990 | default_feature.set_current() |
| 991 | lib = PyDirectory(db, cab, root, "include", "include", "INCLUDE|include") |
| 992 | lib.glob("*.h") |
| 993 | lib.add_file("pyconfig.h", src="../PC/pyconfig.h") |
| 994 | # Add import libraries |
| 995 | lib = PyDirectory(db, cab, root, "PCBuild", "libs", "LIBS|libs") |
| 996 | for f in dlls: |
| 997 | lib.add_file(f.replace('pyd','lib')) |
| 998 | lib.add_file('python%s%s.lib' % (major, minor)) |
Martin v. Löwis | 9fda931 | 2004-12-22 13:41:49 +0000 | [diff] [blame] | 999 | # Add the mingw-format library |
| 1000 | if have_mingw: |
Tim Peters | 5a9fb3c | 2005-01-07 16:01:32 +0000 | [diff] [blame] | 1001 | lib.add_file('libpython%s%s.a' % (major, minor)) |
Martin v. Löwis | e0f780d | 2004-09-01 14:51:06 +0000 | [diff] [blame] | 1002 | if have_tcl: |
| 1003 | # Add Tcl/Tk |
| 1004 | tcldirs = [(root, '../tcltk/lib', 'tcl')] |
| 1005 | tcltk.set_current() |
| 1006 | while tcldirs: |
| 1007 | parent, phys, dir = tcldirs.pop() |
| 1008 | lib = PyDirectory(db, cab, parent, phys, dir, "%s|%s" % (parent.make_short(dir), dir)) |
| 1009 | if not os.path.exists(lib.absolute): |
| 1010 | continue |
| 1011 | for f in os.listdir(lib.absolute): |
| 1012 | if os.path.isdir(os.path.join(lib.absolute, f)): |
| 1013 | tcldirs.append((lib, f, f)) |
| 1014 | else: |
| 1015 | lib.add_file(f) |
Martin v. Löwis | 8ffe9ab | 2004-08-22 13:34:34 +0000 | [diff] [blame] | 1016 | # Add tools |
| 1017 | tools.set_current() |
| 1018 | tooldir = PyDirectory(db, cab, root, "Tools", "Tools", "TOOLS|Tools") |
| 1019 | for f in ['i18n', 'pynche', 'Scripts', 'versioncheck', 'webchecker']: |
| 1020 | lib = PyDirectory(db, cab, tooldir, f, f, "%s|%s" % (tooldir.make_short(f), f)) |
| 1021 | lib.glob("*.py") |
| 1022 | lib.glob("*.pyw", exclude=['pydocgui.pyw']) |
| 1023 | lib.remove_pyc() |
| 1024 | lib.glob("*.txt") |
| 1025 | if f == "pynche": |
| 1026 | x = PyDirectory(db, cab, lib, "X", "X", "X|X") |
| 1027 | x.glob("*.txt") |
Martin v. Löwis | 4d930be | 2004-12-01 21:46:35 +0000 | [diff] [blame] | 1028 | if os.path.exists(os.path.join(lib.absolute, "README")): |
Martin v. Löwis | 8ffe9ab | 2004-08-22 13:34:34 +0000 | [diff] [blame] | 1029 | lib.add_file("README.txt", src="README") |
Martin v. Löwis | 4d930be | 2004-12-01 21:46:35 +0000 | [diff] [blame] | 1030 | if f == 'Scripts': |
Martin v. Löwis | e0f780d | 2004-09-01 14:51:06 +0000 | [diff] [blame] | 1031 | if have_tcl: |
| 1032 | lib.start_component("pydocgui.pyw", tcltk, keyfile="pydocgui.pyw") |
| 1033 | lib.add_file("pydocgui.pyw") |
Martin v. Löwis | 8ffe9ab | 2004-08-22 13:34:34 +0000 | [diff] [blame] | 1034 | # Add documentation |
| 1035 | htmlfiles.set_current() |
| 1036 | lib = PyDirectory(db, cab, root, "Doc", "Doc", "DOC|Doc") |
| 1037 | lib.start_component("documentation", keyfile="Python%s%s.chm" % (major,minor)) |
| 1038 | lib.add_file("Python%s%s.chm" % (major, minor)) |
| 1039 | |
| 1040 | cab.commit(db) |
| 1041 | |
| 1042 | for f in tmpfiles: |
| 1043 | os.unlink(f) |
| 1044 | |
| 1045 | # See "Registry Table", "Component Table" |
| 1046 | def add_registry(db): |
| 1047 | # File extensions, associated with the REGISTRY.def component |
| 1048 | # IDLE verbs depend on the tcltk feature. |
| 1049 | # msidbComponentAttributesRegistryKeyPath = 4 |
| 1050 | # -1 for Root specifies "dependent on ALLUSERS property" |
Martin v. Löwis | e0f780d | 2004-09-01 14:51:06 +0000 | [diff] [blame] | 1051 | tcldata = [] |
| 1052 | if have_tcl: |
| 1053 | tcldata = [ |
Martin v. Löwis | dff68d0 | 2004-09-10 09:20:10 +0000 | [diff] [blame] | 1054 | ("REGISTRY.tcl", msilib.gen_uuid(), "TARGETDIR", 4, None, |
| 1055 | "py.IDLE")] |
Martin v. Löwis | 8ffe9ab | 2004-08-22 13:34:34 +0000 | [diff] [blame] | 1056 | add_data(db, "Component", |
| 1057 | # msidbComponentAttributesRegistryKeyPath = 4 |
| 1058 | [("REGISTRY", msilib.gen_uuid(), "TARGETDIR", 4, None, |
| 1059 | "InstallPath"), |
Martin v. Löwis | 141f41a | 2005-03-15 00:39:40 +0000 | [diff] [blame] | 1060 | ("REGISTRY.doc", msilib.gen_uuid(), "TARGETDIR", 4, None, |
| 1061 | "Documentation"), |
Martin v. Löwis | 8ffe9ab | 2004-08-22 13:34:34 +0000 | [diff] [blame] | 1062 | ("REGISTRY.def", msilib.gen_uuid(), "TARGETDIR", 4, |
Martin v. Löwis | e0f780d | 2004-09-01 14:51:06 +0000 | [diff] [blame] | 1063 | None, None)] + tcldata) |
Martin v. Löwis | 8ffe9ab | 2004-08-22 13:34:34 +0000 | [diff] [blame] | 1064 | # See "FeatureComponents Table". |
| 1065 | # The association between TclTk and pythonw.exe is necessary to make ICE59 |
| 1066 | # happy, because the installer otherwise believes that the IDLE and PyDoc |
| 1067 | # shortcuts might get installed without pythonw.exe being install. This |
| 1068 | # is not true, since installing TclTk will install the default feature, which |
| 1069 | # will cause pythonw.exe to be installed. |
Martin v. Löwis | dff68d0 | 2004-09-10 09:20:10 +0000 | [diff] [blame] | 1070 | # REGISTRY.tcl is not associated with any feature, as it will be requested |
| 1071 | # through a custom action |
Martin v. Löwis | e0f780d | 2004-09-01 14:51:06 +0000 | [diff] [blame] | 1072 | tcldata = [] |
| 1073 | if have_tcl: |
Martin v. Löwis | dff68d0 | 2004-09-10 09:20:10 +0000 | [diff] [blame] | 1074 | tcldata = [(tcltk.id, "pythonw.exe")] |
Martin v. Löwis | 8ffe9ab | 2004-08-22 13:34:34 +0000 | [diff] [blame] | 1075 | add_data(db, "FeatureComponents", |
| 1076 | [(default_feature.id, "REGISTRY"), |
Martin v. Löwis | 141f41a | 2005-03-15 00:39:40 +0000 | [diff] [blame] | 1077 | (htmlfiles.id, "REGISTRY.doc"), |
Martin v. Löwis | e0f780d | 2004-09-01 14:51:06 +0000 | [diff] [blame] | 1078 | (ext_feature.id, "REGISTRY.def")] + |
| 1079 | tcldata |
| 1080 | ) |
Martin v. Löwis | dff68d0 | 2004-09-10 09:20:10 +0000 | [diff] [blame] | 1081 | # Extensions are not advertised. For advertised extensions, |
| 1082 | # we would need separate binaries that install along with the |
| 1083 | # extension. |
Martin v. Löwis | 8ffe9ab | 2004-08-22 13:34:34 +0000 | [diff] [blame] | 1084 | pat = r"Software\Classes\%sPython.%sFile\shell\%s\command" |
| 1085 | ewi = "Edit with IDLE" |
| 1086 | pat2 = r"Software\Classes\%sPython.%sFile\DefaultIcon" |
| 1087 | pat3 = r"Software\Classes\%sPython.%sFile" |
Martin v. Löwis | eac02e6 | 2004-11-18 08:00:33 +0000 | [diff] [blame] | 1088 | tcl_verbs = [] |
| 1089 | if have_tcl: |
| 1090 | tcl_verbs=[ |
| 1091 | ("py.IDLE", -1, pat % (testprefix, "", ewi), "", |
| 1092 | r'"[TARGETDIR]pythonw.exe" "[TARGETDIR]Lib\idlelib\idle.pyw" -n -e "%1"', |
| 1093 | "REGISTRY.tcl"), |
| 1094 | ("pyw.IDLE", -1, pat % (testprefix, "NoCon", ewi), "", |
| 1095 | r'"[TARGETDIR]pythonw.exe" "[TARGETDIR]Lib\idlelib\idle.pyw" -n -e "%1"', |
| 1096 | "REGISTRY.tcl"), |
| 1097 | ] |
Martin v. Löwis | dff68d0 | 2004-09-10 09:20:10 +0000 | [diff] [blame] | 1098 | add_data(db, "Registry", |
| 1099 | [# Extensions |
| 1100 | ("py.ext", -1, r"Software\Classes\."+ext, "", |
| 1101 | "Python.File", "REGISTRY.def"), |
| 1102 | ("pyw.ext", -1, r"Software\Classes\."+ext+'w', "", |
| 1103 | "Python.NoConFile", "REGISTRY.def"), |
| 1104 | ("pyc.ext", -1, r"Software\Classes\."+ext+'c', "", |
| 1105 | "Python.CompiledFile", "REGISTRY.def"), |
| 1106 | ("pyo.ext", -1, r"Software\Classes\."+ext+'o', "", |
| 1107 | "Python.CompiledFile", "REGISTRY.def"), |
| 1108 | # MIME types |
| 1109 | ("py.mime", -1, r"Software\Classes\."+ext, "Content Type", |
| 1110 | "text/plain", "REGISTRY.def"), |
| 1111 | ("pyw.mime", -1, r"Software\Classes\."+ext+'w', "Content Type", |
| 1112 | "text/plain", "REGISTRY.def"), |
| 1113 | #Verbs |
| 1114 | ("py.open", -1, pat % (testprefix, "", "open"), "", |
| 1115 | r'"[TARGETDIR]python.exe" "%1" %*', "REGISTRY.def"), |
| 1116 | ("pyw.open", -1, pat % (testprefix, "NoCon", "open"), "", |
| 1117 | r'"[TARGETDIR]pythonw.exe" "%1" %*', "REGISTRY.def"), |
| 1118 | ("pyc.open", -1, pat % (testprefix, "Compiled", "open"), "", |
| 1119 | r'"[TARGETDIR]python.exe" "%1" %*', "REGISTRY.def"), |
Martin v. Löwis | eac02e6 | 2004-11-18 08:00:33 +0000 | [diff] [blame] | 1120 | ] + tcl_verbs + [ |
Martin v. Löwis | dff68d0 | 2004-09-10 09:20:10 +0000 | [diff] [blame] | 1121 | #Icons |
| 1122 | ("py.icon", -1, pat2 % (testprefix, ""), "", |
Martin v. Löwis | 1319bb1 | 2006-05-12 13:57:36 +0000 | [diff] [blame] | 1123 | r'[DLLs]py.ico', "REGISTRY.def"), |
Martin v. Löwis | dff68d0 | 2004-09-10 09:20:10 +0000 | [diff] [blame] | 1124 | ("pyw.icon", -1, pat2 % (testprefix, "NoCon"), "", |
Martin v. Löwis | 1319bb1 | 2006-05-12 13:57:36 +0000 | [diff] [blame] | 1125 | r'[DLLs]py.ico', "REGISTRY.def"), |
Martin v. Löwis | dff68d0 | 2004-09-10 09:20:10 +0000 | [diff] [blame] | 1126 | ("pyc.icon", -1, pat2 % (testprefix, "Compiled"), "", |
Martin v. Löwis | 1319bb1 | 2006-05-12 13:57:36 +0000 | [diff] [blame] | 1127 | r'[DLLs]pyc.ico', "REGISTRY.def"), |
Martin v. Löwis | dff68d0 | 2004-09-10 09:20:10 +0000 | [diff] [blame] | 1128 | # Descriptions |
| 1129 | ("py.txt", -1, pat3 % (testprefix, ""), "", |
| 1130 | "Python File", "REGISTRY.def"), |
| 1131 | ("pyw.txt", -1, pat3 % (testprefix, "NoCon"), "", |
| 1132 | "Python File (no console)", "REGISTRY.def"), |
| 1133 | ("pyc.txt", -1, pat3 % (testprefix, "Compiled"), "", |
| 1134 | "Compiled Python File", "REGISTRY.def"), |
Martin v. Löwis | 8ffe9ab | 2004-08-22 13:34:34 +0000 | [diff] [blame] | 1135 | ]) |
Tim Peters | 66cb018 | 2004-08-26 05:23:19 +0000 | [diff] [blame] | 1136 | |
Martin v. Löwis | 8ffe9ab | 2004-08-22 13:34:34 +0000 | [diff] [blame] | 1137 | # Registry keys |
| 1138 | prefix = r"Software\%sPython\PythonCore\%s" % (testprefix, short_version) |
| 1139 | add_data(db, "Registry", |
| 1140 | [("InstallPath", -1, prefix+r"\InstallPath", "", "[TARGETDIR]", "REGISTRY"), |
| 1141 | ("InstallGroup", -1, prefix+r"\InstallPath\InstallGroup", "", |
| 1142 | "Python %s" % short_version, "REGISTRY"), |
| 1143 | ("PythonPath", -1, prefix+r"\PythonPath", "", |
Martin v. Löwis | f13337d | 2004-09-19 18:36:45 +0000 | [diff] [blame] | 1144 | r"[TARGETDIR]Lib;[TARGETDIR]DLLs;[TARGETDIR]Lib\lib-tk", "REGISTRY"), |
Martin v. Löwis | 8ffe9ab | 2004-08-22 13:34:34 +0000 | [diff] [blame] | 1145 | ("Documentation", -1, prefix+r"\Help\Main Python Documentation", "", |
Martin v. Löwis | 141f41a | 2005-03-15 00:39:40 +0000 | [diff] [blame] | 1146 | r"[TARGETDIR]Doc\Python%s%s.chm" % (major, minor), "REGISTRY.doc"), |
Martin v. Löwis | 8ffe9ab | 2004-08-22 13:34:34 +0000 | [diff] [blame] | 1147 | ("Modules", -1, prefix+r"\Modules", "+", None, "REGISTRY"), |
| 1148 | ("AppPaths", -1, r"Software\Microsoft\Windows\CurrentVersion\App Paths\Python.exe", |
| 1149 | "", r"[TARGETDIR]Python.exe", "REGISTRY.def") |
| 1150 | ]) |
| 1151 | # Shortcuts, see "Shortcut Table" |
| 1152 | add_data(db, "Directory", |
| 1153 | [("ProgramMenuFolder", "TARGETDIR", "."), |
| 1154 | ("MenuDir", "ProgramMenuFolder", "PY%s%s|%sPython %s.%s" % (major,minor,testprefix,major,minor))]) |
| 1155 | add_data(db, "RemoveFile", |
| 1156 | [("MenuDir", "TARGETDIR", None, "MenuDir", 2)]) |
Martin v. Löwis | e0f780d | 2004-09-01 14:51:06 +0000 | [diff] [blame] | 1157 | tcltkshortcuts = [] |
| 1158 | if have_tcl: |
| 1159 | tcltkshortcuts = [ |
Martin v. Löwis | 8ffe9ab | 2004-08-22 13:34:34 +0000 | [diff] [blame] | 1160 | ("IDLE", "MenuDir", "IDLE|IDLE (Python GUI)", "pythonw.exe", |
Martin v. Löwis | ac191da | 2004-12-22 12:55:44 +0000 | [diff] [blame] | 1161 | tcltk.id, r'"[TARGETDIR]Lib\idlelib\idle.pyw"', None, None, "python_icon.exe", 0, None, "TARGETDIR"), |
Martin v. Löwis | 8ffe9ab | 2004-08-22 13:34:34 +0000 | [diff] [blame] | 1162 | ("PyDoc", "MenuDir", "MODDOCS|Module Docs", "pythonw.exe", |
Martin v. Löwis | ac191da | 2004-12-22 12:55:44 +0000 | [diff] [blame] | 1163 | tcltk.id, r'"[TARGETDIR]Tools\scripts\pydocgui.pyw"', None, None, "python_icon.exe", 0, None, "TARGETDIR"), |
Martin v. Löwis | e0f780d | 2004-09-01 14:51:06 +0000 | [diff] [blame] | 1164 | ] |
| 1165 | add_data(db, "Shortcut", |
| 1166 | tcltkshortcuts + |
| 1167 | [# Advertised shortcuts: targets are features, not files |
Martin v. Löwis | 8ffe9ab | 2004-08-22 13:34:34 +0000 | [diff] [blame] | 1168 | ("Python", "MenuDir", "PYTHON|Python (command line)", "python.exe", |
| 1169 | default_feature.id, None, None, None, "python_icon.exe", 2, None, "TARGETDIR"), |
Martin v. Löwis | 141f41a | 2005-03-15 00:39:40 +0000 | [diff] [blame] | 1170 | # Advertising the Manual breaks on (some?) Win98, and the shortcut lacks an |
| 1171 | # icon first. |
| 1172 | #("Manual", "MenuDir", "MANUAL|Python Manuals", "documentation", |
| 1173 | # htmlfiles.id, None, None, None, None, None, None, None), |
Martin v. Löwis | 8ffe9ab | 2004-08-22 13:34:34 +0000 | [diff] [blame] | 1174 | ## Non-advertised shortcuts: must be associated with a registry component |
Martin v. Löwis | 141f41a | 2005-03-15 00:39:40 +0000 | [diff] [blame] | 1175 | ("Manual", "MenuDir", "MANUAL|Python Manuals", "REGISTRY.doc", |
| 1176 | "[#Python%s%s.chm]" % (major,minor), None, |
| 1177 | None, None, None, None, None, None), |
Martin v. Löwis | 8ffe9ab | 2004-08-22 13:34:34 +0000 | [diff] [blame] | 1178 | ("Uninstall", "MenuDir", "UNINST|Uninstall Python", "REGISTRY", |
| 1179 | SystemFolderName+"msiexec", "/x%s" % product_code, |
| 1180 | None, None, None, None, None, None), |
| 1181 | ]) |
| 1182 | db.Commit() |
| 1183 | |
| 1184 | db = build_database() |
| 1185 | try: |
| 1186 | add_features(db) |
| 1187 | add_ui(db) |
| 1188 | add_files(db) |
| 1189 | add_registry(db) |
| 1190 | remove_old_versions(db) |
| 1191 | db.Commit() |
| 1192 | finally: |
| 1193 | del db |