"""
This module generates .iss file according to the local config of
the current application. Please make sure a file named "local_config.py"
exists in the current directory. Edit local_config.py according to your needs.
"""
import local_config
import os
import string
REG_PROGRAM = """{app}\MYPROG.EXE"" ""%1"""
APPLICATION = str(local_config.__appname__ )+ '.exe'
AppName = str(local_config.__appname__ )
AppVerName = str(local_config.__appname__ )+'-'+ str(local_config.__version__)
Dev = ''
if AppVerName.lower().count('dev') > 0:
Dev = '-Dev'
AppPublisher = local_config._copyright
AppPublisherURL = local_config._homepage
AppSupportURL = local_config._homepage
AppUpdatesURL = local_config._homepage
ChangesEnvironment = 'true'
DefaultDirName = os.path.join("{pf}" , AppName+Dev)
DefaultGroupName = os.path.join(local_config.DefaultGroupName, AppVerName)
OutputBaseFilename = local_config.OutputBaseFilename
SetupIconFile = "images\\ball.ico"
LicenseFile = 'license.txt'
DisableProgramGroupPage = 'yes'
Compression = 'lzma'
SolidCompression = 'yes'
PrivilegesRequired = 'none'
INSTALLER_FILE = 'installer'
icon_path = local_config.icon_path
media_path = local_config.media_path
test_path = local_config.test_path
[docs]def find_extension():
"""
Describe the extensions that can be read by the current application
"""
list_data = []
list_app =[]
try:
#(ext, type, name, flags)
from sas.sascalc.dataloader.loader import Loader
wild_cards = Loader().get_wildcards()
for item in wild_cards:
#['All (*.*)|*.*']
file_type, ext = string.split(item, "|*", 1)
if ext.strip() not in ['.*', ''] and ext.strip() not in list_data:
list_data.append((ext, 'string', file_type))
except:
pass
try:
file_type, ext = string.split(local_config.APPLICATION_WLIST, "|*", 1)
if ext.strip() not in ['.', ''] and ext.strip() not in list_app:
list_app.append((ext, 'string', file_type))
except:
pass
try:
for item in local_config.PLUGINS_WLIST:
file_type, ext = string.split(item, "|*", 1)
if ext.strip() not in ['.', ''] and ext.strip() not in list_app:
list_app.append((ext, 'string', file_type))
except:
pass
return list_data, list_app
DATA_EXTENSION, APP_EXTENSION = find_extension()
[docs]def write_registry(data_extension=None, app_extension=None):
"""
create file association for windows.
Allow open file on double click
"""
msg = ""
if data_extension is not None and data_extension:
openwithlist = "OpenWithList\%s" % str(APPLICATION)
msg = "\n\n[Registry]\n"
for (ext, type, _) in data_extension:
list = os.path.join(ext, openwithlist)
msg += """Root: HKCR;\tSubkey: "%s";\t""" % str(list)
msg += """ Flags: %s""" % str('uninsdeletekey noerror')
msg += "\n"
#list the file on right-click
msg += """Root: HKCR; Subkey: "applications\%s\shell\open\command";\t"""\
% str(APPLICATION)
msg += """ValueType: %s; """ % str('string')
msg += """ValueName: "%s";\t""" %str('')
msg += """ValueData: \"""{app}\%s"" ""%s1\"""; \t"""% (str(APPLICATION),
str('%'))
msg += """ Flags: %s""" % str('uninsdeletevalue noerror')
msg += "\n"
user_list = "Software\Classes"
for (ext, type, _) in data_extension:
list = os.path.join(user_list, ext, openwithlist)
msg += """Root: HKCU;\tSubkey: "%s";\t""" % str(list)
msg += """ Flags: %s""" % str('uninsdeletekey noerror')
msg += "\n"
#list the file on right-click
user_list = os.path.join("Software", "Classes", "applications")
msg += """Root: HKCU; Subkey: "%s\%s\shell\open\command";\t"""\
% (str(user_list), str(APPLICATION))
msg += """ValueType: %s; """ % str('string')
msg += """ValueName: "%s";\t""" %str('')
msg += """ValueData: \"""{app}\%s"" ""%s1\"""; \t"""% (str(APPLICATION),
str('%'))
msg += """ Flags: %s""" % str('uninsdeletevalue noerror')
msg += "\n"
if app_extension is not None and app_extension:
for (ext, type, _) in app_extension:
msg += """Root: HKCR;\tSubkey: "%s";\t""" % str(ext)
msg += """ValueType: %s;\t""" % str(type)
#file type empty set the current application as the default
#reader for this file. change the value of file_type to another
#string modify the default reader
file_type = ''
msg += """ValueName: "%s";\t""" % str('')
msg += """ValueData: "{app}\%s";\t""" % str(APPLICATION)
msg += """ Flags: %s""" % str('uninsdeletevalue noerror')
msg += "\n"
msg += """Root: HKCR; Subkey: "{app}\%s";\t""" % str(APPLICATION)
msg += """ValueType: %s; """ % str('string')
msg += """ValueName: "%s";\t""" % str('')
msg += """ValueData: "{app}\%s";\t""" % str("SasView File")
msg += """ Flags: %s \t""" % str("uninsdeletekey noerror")
msg += "\n"
#execute the file on double-click
msg += """Root: HKCR; Subkey: "{app}\%s\shell\open\command";\t""" % str(APPLICATION)
msg += """ValueType: %s; """ % str('string')
msg += """ValueName: "%s";\t""" %str('')
msg += """ValueData: \"""{app}\%s"" ""%s1\""";\t"""% (str(APPLICATION),
str('%'))
msg += """ Flags: %s \t""" % str("uninsdeletevalue noerror")
msg += "\n"
#create default icon
msg += """Root: HKCR; Subkey: "{app}\%s";\t""" % str(SetupIconFile)
msg += """ValueType: %s; """ % str('string')
msg += """ValueName: "%s";\t""" % str('')
msg += """ValueData: "{app}\%s,0";\t""" % str(APPLICATION)
msg += """ Flags: %s \t""" % str("uninsdeletevalue noerror")
msg += "\n"
#SASVIEWPATH
msg += """Root: HKLM; Subkey: "%s";\t""" % str('SYSTEM\CurrentControlSet\Control\Session Manager\Environment')
msg += """ValueType: %s; """ % str('expandsz')
msg += """ValueName: "%s";\t""" % str('SASVIEWPATH')
msg += """ValueData: "{app}";\t"""
msg += """ Flags: %s""" % str('uninsdeletevalue noerror')
msg += "\n"
#PATH
msg += """; Write to PATH (below) is disabled; need more tests\n"""
msg += """;Root: HKCU; Subkey: "%s";\t""" % str('Environment')
msg += """ValueType: %s; """ % str('expandsz')
msg += """ValueName: "%s";\t""" % str('PATH')
msg += """ValueData: "%s;{olddata}";\t""" % str('%SASVIEWPATH%')
msg += """ Check: %s""" % str('NeedsAddPath()')
msg += "\n"
return msg
[docs]def write_language(language=['english'], msfile="compiler:Default.isl"):
"""
define the language of the application
"""
msg = ''
if language:
msg = "\n\n[Languages]\n"
for lang in language:
msg += """Name: "%s";\tMessagesFile: "%s"\n""" % (str(lang),
str(msfile))
return msg
[docs]def write_tasks():
"""
create desktop icon
"""
msg = """\n\n[Tasks]\n"""
msg += """Name: "desktopicon";\tDescription: "{cm:CreateDesktopIcon}";\t"""
msg += """GroupDescription: "{cm:AdditionalIcons}";\tFlags: unchecked\n"""
msg += """Name: "quicklaunchicon"; Description: "{cm:CreateQuickLaunchIcon}";\t"""
msg += """GroupDescription: "{cm:AdditionalIcons}";\n"""
return msg
dist_path = "dist"
[docs]def write_file():
"""
copy some data files
"""
msg = "\n\n[Files]\n"
msg += """Source: "%s\%s";\t""" % (dist_path, str(APPLICATION))
msg += """DestDir: "{app}";\tFlags: ignoreversion\n"""
msg += """Source: "dist\*";\tDestDir: "{app}";\t"""
msg += """Flags: ignoreversion recursesubdirs createallsubdirs\n"""
msg += """Source: "dist\plugin_models\*";\tDestDir: "{userdesktop}\..\.sasview\plugin_models";\t"""
msg += """Flags: recursesubdirs createallsubdirs\n"""
msg += """Source: "dist\compiled_models\*";\tDestDir: "{userdesktop}\..\.sasmodels\compiled_models";\t"""
msg += """Flags: recursesubdirs createallsubdirs\n"""
msg += """Source: "dist\config\custom_config.py";\tDestDir: "{userdesktop}\..\.sasview\config";\t"""
msg += """Flags: recursesubdirs createallsubdirs\n"""
#msg += """Source: "dist\default_categories.json"; DestDir: "{userdesktop}\..\.sasview";\t"""
#msg += """DestName: "categories.json";\n"""
msg += """;\tNOTE: Don't use "Flags: ignoreversion" on any shared system files"""
return msg
[docs]def write_icon():
"""
Create application icon
"""
msg = """\n\n[Icons]\n"""
msg += """Name: "{group}\%s";\t""" % str(AppName)
msg += """Filename: "{app}\%s";\t""" % str(APPLICATION)
msg += """WorkingDir: "{app}"; IconFilename: "{app}\images\\ball.ico" \n"""
msg += """Name: "{group}\{cm:UninstallProgram, %s}";\t""" % str(AppName)
msg += """ Filename: "{uninstallexe}" \n"""
msg += """Name: "{commondesktop}\%s";\t""" % str(AppVerName)
msg += """Filename: "{app}\%s";\t""" % str(APPLICATION)
msg += """Tasks: desktopicon; WorkingDir: "{app}" ; IconFilename: "{app}\images\\ball.ico" \n"""
msg += """Name: "{userappdata}\Microsoft\Internet Explorer\Quick Launch\%s";\t""" % str(AppVerName)
msg += """Filename: "{app}\%s";\t""" % str(APPLICATION)
msg += """Tasks: quicklaunchicon; WorkingDir: "{app}"; IconFilename: "{app}\images\\ball.ico" \n"""
return msg
[docs]def write_run():
"""
execute some file
"""
msg = """\n\n[Run]\n"""
msg += """Filename: "{app}\%s";\t""" % str(APPLICATION)
msg += """Description: "{cm:LaunchProgram, %s}";\t""" %str(AppName)
msg += """Flags: nowait postinstall skipifsilent\n"""
msg += """; Install the Microsoft C++ DLL redistributable package if it is """
msg += """provided and the DLLs are not present on the target system.\n"""
msg += """; Note that the redistributable package is included if the app was """
msg += """built using Python 2.6 or 2.7, but not with 2.5.\n"""
msg += """; Parameter options:\n"""
msg += """; - for silent install use: "/q"\n"""
msg += """; - for silent install with progress bar use: "/qb"\n"""
msg += """; - for silent install with progress bar but disallow """
msg += """cancellation of operation use: "/qb!"\n"""
msg += """; Note that we do not use the postinstall flag as this would """
msg += """display a checkbox and thus require the user to decide what to do.\n"""
msg += """;Filename: "{app}\\vcredist_x86.exe"; Parameters: "/qb!"; """
msg += """WorkingDir: "{tmp}"; StatusMsg: "Installing Microsoft Visual """
msg += """C++ 2008 Redistributable Package ..."; Check: InstallVC90CRT(); """
msg += """Flags: skipifdoesntexist waituntilterminated\n"""
return msg
[docs]def write_dirs():
"""
Define Dir permission
"""
msg = """\n\n[Dirs]\n"""
msg += """Name: "{app}\%s";\t""" % str('')
msg += """Permissions: everyone-modify\t"""
msg += """\n"""
return msg
[docs]def write_code():
"""
Code that checks the existing path and snaviewpath
in the environmental viriables/PATH
"""
msg = """\n\n[Code]\n"""
msg += """function InstallVC90CRT(): Boolean;\n"""
msg += """begin\n"""
msg += """ Result := not DirExists('C:\WINDOWS\WinSxS\\x86_Microsoft.VC90."""
msg += """CRT_1fc8b3b9a1e18e3b_9.0.21022.8_x-ww_d08d0375');\n"""
msg += """end;\n\n"""
msg += """function NeedsAddPath(): boolean;\n"""
msg += """var\n"""
msg += """ oldpath: string;\n"""
msg += """ newpath: string;\n"""
msg += """ pathArr: TArrayOfString;\n"""
msg += """ i: Integer;\n"""
msg += """begin\n"""
msg += """ RegQueryStringValue(HKEY_CURRENT_USER,'Environment',"""
msg += """'PATH', oldpath)\n"""
msg += """ oldpath := oldpath + ';';\n"""
msg += """ newpath := '%SASVIEWPATH%';\n"""
msg += """ i := 0;\n"""
msg += """ while (Pos(';', oldpath) > 0) do begin\n"""
msg += """ SetArrayLength(pathArr, i+1);\n"""
msg += """ pathArr[i] := Copy(oldpath, 0, Pos(';', oldpath)-1);\n"""
msg += """ oldpath := Copy(oldpath, Pos(';', oldpath)+1,"""
msg += """ Length(oldpath));\n"""
msg += """ i := i + 1;\n"""
msg += """ // Check if current directory matches app dir\n"""
msg += """ if newpath = pathArr[i-1] \n"""
msg += """ then begin\n"""
msg += """ Result := False;\n"""
msg += """ exit;\n"""
msg += """ end;\n"""
msg += """ end;\n"""
msg += """ Result := True;\n"""
msg += """end;\n"""
msg += """\n"""
return msg
[docs]def write_uninstalldelete():
"""
Define uninstalldelete
"""
msg = """\n[UninstallDelete]\n"""
msg += """; Delete directories and files that are dynamically created by """
msg += """the application (i.e. at runtime).\n"""
msg += """Type: filesandordirs; Name: "{app}\.matplotlib"\n"""
msg += """Type: files; Name: "{app}\*.*"\n"""
msg += """; The following is a workaround for the case where the """
msg += """application is installed and uninstalled but the\n"""
msg += """;{app} directory is not deleted because it has user files. """
msg += """Then the application is installed into the\n"""
msg += """; existing directory, user files are deleted, and the """
msg += """application is un-installed again. Without the\n"""
msg += """; directive below, {app} will not be deleted because Inno Setup """
msg += """did not create it during the previous\n"""
msg += """; installation.\n"""
msg += """Type: dirifempty; Name: "{app}"\n"""
msg += """\n"""
return msg
[docs]def generate_installer():
"""
"""
TEMPLATE = "\n; Script generated by the Inno Setup Script Wizard\n"
TEMPLATE += "\n; and local_config.py located in this directory.\n "
TEMPLATE += "; SEE THE DOCUMENTATION FOR DETAILS ON CREATING INNO SETUP SCRIPT FILES!"
TEMPLATE += "\n[Setup]\n\n"
TEMPLATE += "ChangesAssociations=%s\n" %str('yes')
TEMPLATE += "AppName=%s\n" % str(AppName)
TEMPLATE += "AppVerName=%s\n" % str(AppVerName)
TEMPLATE += "AppPublisher=%s\n" % str(AppPublisher)
TEMPLATE += "AppPublisherURL=%s\n" % str(AppPublisherURL)
TEMPLATE += "AppSupportURL=%s\n" % str(AppSupportURL)
TEMPLATE += "AppUpdatesURL=%s \n" % str(AppUpdatesURL)
TEMPLATE += "ChangesEnvironment=%s \n" % str(ChangesEnvironment)
TEMPLATE += "DefaultDirName=%s\n" % str(DefaultDirName)
TEMPLATE += "DefaultGroupName=%s\n" % str(DefaultGroupName)
TEMPLATE += "DisableProgramGroupPage=%s\n" % str(DisableProgramGroupPage)
TEMPLATE += "LicenseFile=%s\n" % str(LicenseFile)
TEMPLATE += "OutputBaseFilename=%s\n" % str(OutputBaseFilename)
TEMPLATE += "SetupIconFile=%s\n" % str(SetupIconFile)
TEMPLATE += "Compression=%s\n" % str(Compression)
TEMPLATE += "SolidCompression=%s\n" % str(SolidCompression)
TEMPLATE += "PrivilegesRequired=%s\n" % str(PrivilegesRequired)
TEMPLATE += "UsePreviousAppDir=no\n"
TEMPLATE += write_registry(data_extension=DATA_EXTENSION,
app_extension=APP_EXTENSION)
TEMPLATE += write_language()
TEMPLATE += write_tasks()
TEMPLATE += write_file()
TEMPLATE += write_icon()
TEMPLATE += write_run()
TEMPLATE += write_dirs()
TEMPLATE += write_code()
TEMPLATE += write_uninstalldelete()
path = '%s.iss' % str(INSTALLER_FILE)
f = open(path,'w')
f.write(TEMPLATE)
f.close()
print "Generate Inno setup installer script complete"
print "A new file %s.iss should be created.Please refresh your directory" % str(INSTALLER_FILE)
if __name__ == "__main__":
generate_installer()