Удалите другой MSI при установке

У меня есть базовый проект MSI. При установке мне нужно удалить другой продукт MSI, который теперь интегрирован в наше основное приложение. Я пытался использовать сценарии обновления и рассматривать это как серьезное обновление. Однако это не сработало из-за несовпадения кодов обновления, как мне кажется.

Затем я также сделал настраиваемое действие, которое запускало msiexec.exe после CostFinalize (я думаю, это было указано в справке Installshield). Это работало отлично, пока я не установил систему, в которой не было установщика, который я хотел удалить. Мой установщик не сработает, если не будет установлен другой устаревший продукт. Я попытался поставить условие для настраиваемого действия, установленного системным поиском, но похоже, что системный поиск ограничен в функциональности. Я не могу просто проверить ключ reg и установить логическое свойство.

Любые идеи?


person Web    schedule 09.03.2010    source источник


Ответы (2)


Несколько вещей, которые следует учитывать

1) UpgradeTable (FindRelatedProducts / RemoveExisting Products) может использоваться для удаления ProductCodes, связанных с UpgradeCode другого продукта.

2) Если память не исправна, MSI не удалит продукт для пользователя во время установки на компьютере (или наоборот). Контекст должен быть таким же.

3) Последовательность пользовательского интерфейса не запускается во время автоматической установки.

4) Вы не можете запустить msiexec из последовательности выполнения, потому что существует общесистемный мьютекс, состоящий только из одной последовательности выполнения для каждой машины.

5) Если вы планируете в пользовательском интерфейсе (я уже сказал вам, что этого не следует делать, поскольку он не запускается во время тихой установки), есть еще один мьютекс, который говорит только 1 пользовательский интерфейс для каждого процесса.

Если вы переходите от одного пользователя к другому или от одного компьютера к другому, я считаю разумным, что вы должны иметь возможность делать то, что хотите, используя элементы / строки таблицы Upgrade без написания настраиваемых действий. В противном случае вам понадобится загрузчик в стиле setup.exe для обработки удаления до входа в мир msiexec.

person Christopher Painter    schedule 10.03.2010
comment
Спасибо за информацию. Я смог выполнить то, что мне было нужно, используя элементы обновления, как вы предложили. - person Web; 10.03.2010

Я добился этого в InstallShield 2013, используя собственный InstallScript. Сценарий выполняется с помощью настраиваемого действия в последовательности пользовательского интерфейса, но я поместил его после диалогового окна «SetupProgress», то есть перед «Execute Action», а не после CostFinalize (как сказано в документации). Я добавил в действие условие "НЕ установлено". Если вы разместите это в предложенном порядке, удаление начнется сразу после инициализации установщика. Если вы переместите его туда, где я это сделал, он не запустится, пока пользователь не нажмет последнюю кнопку «Установить сейчас».

Причина включения этого в последовательность пользовательского интерфейса состоит в том, чтобы обойти один установщик msi (или деинсталлятор) за раз. Это просто не работает в последовательности выполнения из-за этого ограничения.

Основная слабость этого метода заключается в том, что, как заявил Кристофер, он не будет работать при автоматической установке (это также можно найти в документации IS). Однако это официальное средство для достижения этой цели. (проверьте: http://helpnet.installshield.com/installshield16helplib/IHelpCustomActionMSIExec.htm Если вы можете смириться с этим (поскольку тихая установка, как правило, является частным случаем), тогда это работает нормально.

Как также сказал Крис, вы не можете запустить пользовательский интерфейс деинсталлятора, пока работает основной пользовательский интерфейс, но это не проблема с моим скриптом, потому что он добавляет переключатель командной строки для запуска деинсталлятора без пользовательского интерфейса (т. Е. Незаметно).

Мой сценарий также избегает необходимости знать руководство приложения, которое вы хотите удалить. Вот сценарий для привязки к настраиваемому действию (UninstallPriorVersions - функция точки входа):

////////////////////////////////////////////////////////////////////////////////
    //                                                                            
    //  This template script provides the code necessary to build an entry-point 
    //  function to be called in an InstallScript custom action. 
    //                                                                            
    //                                                                            
    //    File Name:  Setup.rul                                                   
    //                                                                            
    //  Description:  InstallShield script                                        
    //
    ////////////////////////////////////////////////////////////////////////////////

    // Include Ifx.h for built-in InstallScript function prototypes, for Windows 
    // Installer API function prototypes and constants, and to declare code for 
    // the OnBegin and OnEnd events.
    #include "ifx.h"

    // The keyword export identifies MyFunction() as an entry-point function.
    // The argument it accepts must be a handle to the Installer database.

    export prototype UninstallPriorVersions(HWND);

    // To Do:  Declare global variables, define constants, and prototype user-
    //         defined and DLL functions here.

    prototype NUMBER UninstallApplicationByName( STRING );
    prototype NUMBER GetUninstallCmdLine( STRING, BOOL, BYREF STRING );
    prototype STRING GetUninstallKey( STRING );
    prototype NUMBER RegDBGetSubKeyNameContainingValue( NUMBER, STRING, STRING, STRING, BYREF STRING );

    // To Do:  Create a custom action for this entry-point function:
    // 1.  Right-click on "Custom Actions" in the Sequences/Actions view.
    // 2.  Select "Custom Action Wizard" from the context menu.
    // 3.  Proceed through the wizard and give the custom action a unique name.
    // 4.  Select "Run InstallScript code" for the custom action type, and in
    //     the next panel select "MyFunction" (or the new name of the entry-
    //     point function) for the source.
    // 5.  Click Next, accepting the default selections until the wizard
    //     creates the custom action.
    //
    // Once you have made a custom action, you must execute it in your setup by
    // inserting it into a sequence or making it the result of a dialog's
    // control event.


    ///////////////////////////////////////////////////////////////////////////////
    //                                                                           
    // Function:  UninstallPriorVersions
    //                                                                           
    //  Purpose:  Uninstall prior versions of this application
    //                                                                           
    ///////////////////////////////////////////////////////////////////////////////
    function UninstallPriorVersions(hMSI)
    begin

        UninstallApplicationByName( "The Name Of Some App" );           

    end;


    ///////////////////////////////////////////////////////////////////////////////
    //                                                                           
    // Function: UninstallApplicationByName
    //                                                                           
    // Purpose: Uninstall an application (without knowing the guid)
    //                        
    // Returns: (UninstCmdLine is assigned a value by referrence)
    //      >= ISERR_SUCCESS    The function successfully got the command line.
    //      < ISERR_SUCCESS     The function failed to get the command line.
    //
    ///////////////////////////////////////////////////////////////////////////////
    function NUMBER UninstallApplicationByName( AppName )
        NUMBER nReturn;
        STRING UninstCmdLine;
    begin           

        nReturn = GetUninstallCmdLine( AppName, TRUE, UninstCmdLine );  
        if( nReturn < ISERR_SUCCESS ) then
            return nReturn;
        endif;

        if( LaunchAppAndWait( "", UninstCmdLine, LAAW_OPTION_WAIT) = 0 ) then 
            return ISERR_SUCCESS;
        else
            return ISERR_SUCCESS-1;
        endif; 

    end;


    ///////////////////////////////////////////////////////////////////////////////
    //                                                                           
    // Function: GetUninstallCmdLine 
    //                                                                           
    // Purpose: Get the command line statement to uninstall an application 
    //                        
    // Returns: (UninstCmdLine is assigned a value by referrence)
    //      >= ISERR_SUCCESS    The function successfully got the command line.
    //      < ISERR_SUCCESS     The function failed to get the command line.
    //
    ///////////////////////////////////////////////////////////////////////////////
    function NUMBER GetUninstallCmdLine( AppName, Silent, UninstCmdLine )
        NUMBER nReturn;
    begin           

        nReturn = RegDBGetUninstCmdLine ( GetUninstallKey( AppName ), UninstCmdLine );
        if( nReturn < ISERR_SUCCESS ) then
            return nReturn;
        endif;

        if( Silent && StrFind( UninstCmdLine, "MsiExec.exe" ) >= 0 )then
            UninstCmdLine = UninstCmdLine + " /qn"; 
        endif; 

        return nReturn;
    end;


    ///////////////////////////////////////////////////////////////////////////////
    //                                                                           
    // Function: GetUninstallKey
    //                                                                           
    // Purpose:  Find the uninstall key in the registry for an application looked up by name
    //      
    // Returns: The uninstall key (i.e. the guid or a fall back value)
    //                                                                       
    ///////////////////////////////////////////////////////////////////////////////
    function STRING GetUninstallKey( AppName )
        STRING guid;
        STRING Key64, Key32, ValueName;
    begin

        Key64 = "SOFTWARE\\Microsoft\\Windows\\CurrentVersion\\Uninstall";
        Key32 = "SOFTWARE\\Wow6432Node\\\Microsoft\\Windows\\CurrentVersion\\Uninstall"; 
        ValueName = "DisplayName";

        if( RegDBGetSubKeyNameContainingValue( HKEY_LOCAL_MACHINE, Key64, ValueName, AppName, guid ) = 0 ) then
            return guid; // return 64 bit GUID
        endif;

        if( RegDBGetSubKeyNameContainingValue( HKEY_LOCAL_MACHINE, Key32, ValueName, AppName, guid ) = 0 ) then
            return guid; // return 32 bit GUID
        endif;

        return AppName; // return old style uninstall key (fall back value)

    end;


    ///////////////////////////////////////////////////////////////////////////////
    //                                                                           
    // Function: RegDBGetSubKeyNameContainingValue
    //                                                                           
    // Purpose:  Find a registry sub key containing a given value.
    //           Return the NAME of the subkey (NOT the entire key path)
    //
    // Returns: (SubKeyName is assigned a value by referrence)
    //      = 0     A sub key name was found with a matching value
    //      != 0    Failed to find a sub key with a matching value
    //                                                                           
    ///////////////////////////////////////////////////////////////////////////////
    function NUMBER RegDBGetSubKeyNameContainingValue( nRootKey, Key, ValueName, Value, SubKeyName )
        STRING SearchSubKey, SubKey, svValue;
        NUMBER nResult, nType, nvSize;
        LIST   listSubKeys;
    begin

        SubKeyName = "";

        listSubKeys  = ListCreate(STRINGLIST);
        if (listSubKeys = LIST_NULL) then
            MessageBox ("Unable to create necessary list.", SEVERE);
            abort;
        endif;

        RegDBSetDefaultRoot( nRootKey );

        if (RegDBQueryKey( Key, REGDB_KEYS, listSubKeys ) = 0) then    
            nResult = ListGetFirstString (listSubKeys, SubKey); 
            while (nResult != END_OF_LIST) 
                SearchSubKey = Key + "\\" + SubKey;
                nType = REGDB_STRING;
                if (RegDBGetKeyValueEx (SearchSubKey, ValueName, nType, svValue, nvSize) = 0) then
                    if( svValue = Value ) then              
                        SubKeyName = SubKey;     
                        nResult = END_OF_LIST;
                    endif;
                endif;      
                if( nResult != END_OF_LIST ) then                       
                    nResult = ListGetNextString (listSubKeys, SubKey); 
                endif;
            endwhile; 
        endif;

        ListDestroy (listSubKeys );

        if ( SubKeyName = "" ) then
            return 1;
        else
            return 0;
        endif;

    end;
person BuvinJ    schedule 13.02.2015