Подтвердить что ты не робот

Как включить номер версии в VS Setup Project output filename

Есть ли способ включить номер версии как часть имени output.msi в проекте установки VS2008?

Я хотел бы, например, получить выходной файл, называемый "myinstaller-1.0.13.msi", где часть версии автоматически устанавливается на основе номера версии, которую я вложил в свойства проекта развертывания.

4b9b3361

Ответ 1

Не уверен, что вы все еще требуете этого или нет, но хотите ответить на этот вопрос, поскольку мы выполняли аналогичную операцию в событии postbuild. Что касается исследований, которые я сделал, это невозможно установить имя файла, как вы хотите внутренне через процесс установки.

Вы можете сделать это другим способом, назвав выходной файл внешним приложением в событии post build.

Вот что вы можете сделать:

В событии post build →

[MsiRenamerAppPath]\MsiRenamer.exe "$ (BuildOutputPath)"

Создайте приложение, которое переименует файл msi с номером версии из проекта развертывания. Ниже приведен код, используемый для приложения. Это должно выполнить ваше требование, я думаю.

Получение кода свойств msi используется статья alteridem

class MsiRenamer
  {
    static void Main(string[] args)
    {
      string inputFile;
      string productName = "[ProductName]";

      if (args.Length == 0)
      {
        Console.WriteLine("Enter MSI file:");
        inputFile = Console.ReadLine();
      }
      else
      {
        inputFile = args[0];
      }

      try
      {
        string version;

        if (inputFile.EndsWith(".msi", StringComparison.OrdinalIgnoreCase))
        {
          // Read the MSI property
          version = GetMsiProperty(inputFile, "ProductVersion");
          productName = GetMsiProperty(inputFile, "ProductName");
        }
        else
        {
          return;
        }
        // Edit: MarkLakata: .msi extension is added back to filename
        File.Copy(inputFile, string.Format("{0} {1}.msi", productName, version));
        File.Delete(inputFile);
      }
      catch (Exception ex)
      {
        Console.WriteLine(ex.Message);
      }
    }

    static string GetMsiProperty(string msiFile, string property)
    {
      string retVal = string.Empty;

      // Create an Installer instance  
      Type classType = Type.GetTypeFromProgID("WindowsInstaller.Installer");
      Object installerObj = Activator.CreateInstance(classType);
      Installer installer = installerObj as Installer;

      // Open the msi file for reading  
      // 0 - Read, 1 - Read/Write  
      Database database = installer.OpenDatabase(msiFile, 0);

      // Fetch the requested property  
      string sql = String.Format(
          "SELECT Value FROM Property WHERE Property='{0}'", property);
      View view = database.OpenView(sql);
      view.Execute(null);

      // Read in the fetched record  
      Record record = view.Fetch();
      if (record != null)
      {
        retVal = record.get_StringData(1);
        System.Runtime.InteropServices.Marshal.FinalReleaseComObject(record);
      }
      view.Close();
      System.Runtime.InteropServices.Marshal.FinalReleaseComObject(view);
      System.Runtime.InteropServices.Marshal.FinalReleaseComObject(database);

      return retVal;
    }
  }

Ответ 2

Я не хотел использовать выше описанный метод .exe, и у меня было немного свободного времени, поэтому я начал искать. Я использую VS 2008 для Windows 7 64 бит. Когда у меня есть проект установки, позвоните на MySetup, все детали проекта можно найти в файле $(ProjectDir) MySetup.vdproj.

Версия продукта будет найдена в одной строке в этом файле в форме

ProductVersion="8:1.0.0"

Теперь в проекте установки есть событие post-build. Если вы выберете проект установки и нажмите F4, вы получите совершенно другой набор свойств, когда вы щелкните правой кнопкой мыши и выберите свойства. После удара F4 вы увидите, что один из них - PostBuildEvent. Опять же, предполагая, что проект установки называется MySetup, следующее будет указывать имя .msi для включения даты и версии

set datevar=%DATE:~6,4%%DATE:~3,2%%DATE:~0,2%
findstr /v PostBuildEvent $(ProjectDir)MySetup.vdproj | findstr ProductVersion >$(ProjectDir)version.txt
set /p var=<$(ProjectDir)version.txt
set var=%var:"=%
set var=%var: =%
set var=%var:.=_%
for /f "tokens=1,2 delims=:" %%i in ("%var%") do @echo %%j >$(ProjectDir)version.txt
set /p realvar=<$(ProjectDir)version.txt
rename "$(ProjectDir)$(Configuration)\MySetup.msi" "MySetup-%datevar%-%realvar%.msi"

Я рассмотрю вышесказанное.

datevar - текущая дата в форме YYYYMMDD.

Линия findstr проходит через MySetup.vdproj, удаляет любую строку с PostBuildEvent внутри, затем возвращает единственную строку, оставленную с помощью productVersion, и выводит ее в файл. Затем мы удаляем кавычки, пробелы, поворачиваем точки в символы подчеркивания.

Строка for разделяет оставшуюся строку на двоеточие и берет вторую часть и снова выводит ее в файл.

Затем мы устанавливаем значение realvar в значение, оставшееся в файле, и переименем MySetup.msi, чтобы указать дату и версию.

Итак, учитывая ProductVersion выше, если это было 27 марта 2012 года, файл будет переименован в

MySetup-20120327-1_0_0.msi

Очевидно, что с помощью этого метода вы можете получить ЛЮБОЙ из переменных в файле vdproj и включить их в свое имя выходного файла, и нам не нужно создавать какие-либо дополнительные .exe-программы для этого.

НТН

Ответ 3

Та же концепция, что и Джим Гримметт, но с меньшими зависимостями:

FOR /F "tokens=2 delims== " %%V IN ('FINDSTR /B /R /C:" *\"ProductVersion\"" "$(ProjectDir)MySetupProjectName.vdproj"') DO FOR %%I IN ("$(BuiltOuputPath)") DO REN "$(BuiltOuputPath)" "%%~nI-%%~nxV%%~xI"

Некоторые примечания:

MySetupProjectName.vdproj следует изменить на имя вашего файла проекта. Забывание изменить это приводит к ошибке сборки: 'PostBuildEvent' failed with error code '1' и окно Output показывает, какой файл FINDSTR не может быть открыт.

Пошаговое описание:

FINDSTR /B /R /C:" *\"ProductVersion\"" $(ProjectDir)MySetupProjectName.vdproj

  • Это находит строку "ProductVersion" = "8:x.y.z.etc" в файле проекта.

FOR /F "tokens=2 delims== " %%V IN (...) DO ... %%~nxV ...

  • Это используется для анализа части x.y.z.etc из приведенного выше результата.

$(BuiltOuputPath)

  • Это исходный путь вывода в соответствии с тем, что он говорит в командной строке "Макросы" после сборки.

FOR %%I IN (...) DO ... %%~nI-%%~nxV%%~xI

  • Это используется для преобразования строки foo.msi в foo-x.y.z.etc.msi.

REN "$(BuiltOuputPath)" ...

  • Это просто переименовывает путь вывода к новому имени.

FOR ... DO FOR .. DO REN ...

  • Он записывается на одной строке, подобной этой, так что ошибка вдоль пути полностью разбивает сборку.

Ответ 4

Если вы используете проект WIX (в отличие от проекта установки и развертывания VS), в этой статье объясняется, как добиться того, что вы после того, как.

Ответ 5

using System;
using System.Collections.Generic;
using System.Linq;
using System.Text;
using System.IO;
using WindowsInstaller;


// cscript //nologo "$(ProjectDir)WiRunSql.vbs" "$(BuiltOuputPath)" "UPDATE `Property` SET `Property`.`Value`='4.0.0.1' WHERE `Property`='ProductVersion'"
// "SELECT `Property`.`ProductVersion` FROM `Property` WHERE `Property`.`Property` = 'ProductVersion'"

/* 
 * That a .NET wrapper generated by tlbimp.exe, wrapping the ActiveX component c:\windows\system32\msi.dll.  
 * You can let the IDE make one for you with Project + Add Reference, COM tab, 
 * select "Microsoft Windows Installer Object Library". 
 */
namespace PostBuildEventModifyMSI
{
    /* Post build event fro Rename MSI file.
     * $(SolutionDir)PostBuildEventModifyMSI\bin\Debug\PostBuildEventModifyMSI.exe "$(SolutionDir)TestWebApplicationSetup\Debug\TestWebApplicationSetup.msi"
     */

    [System.Runtime.InteropServices.ComImport(), System.Runtime.InteropServices.Guid("000C1090-0000-0000-C000-000000000046")]
    class Installer { }
    class Program
    {
        static void Main(string[] args)
        {
            #region New code.

            string msiFilePath = string.Empty;
            if (args.Length == 0)
            {
                Console.WriteLine("Enter MSI file complete path:");
                msiFilePath = Console.ReadLine();
            }
            else
            {
                Console.WriteLine("Argument Received args[0]: " + args[0]);
                msiFilePath = args[0];
            }

            StringBuilder sb = new StringBuilder();
            string[] words = msiFilePath.Split('\\');
            foreach (string word in words)
            {
                sb.Append(word + '\\');

                if (word.Contains("Debug"))
                {
                    break;
                }
                else
                {

                }
            }

            // Open a view on the Property table for the Label property 
            //UPDATE Property set Value = '2.06.36' where Property = 'ProductVersion'
            Program p = new Program();
            string version = p.GetMsiVersionProperty(msiFilePath, "ProductVersion");
            string productName = p.GetMsiVersionProperty(msiFilePath, "ProductName");

            string newMSIpath = sb.ToString() + string.Format("{0}_{1}.msi", productName, version);
            Console.WriteLine("Original MSI File Path: " + msiFilePath);
            Console.WriteLine("New MSI File Path: " + newMSIpath);


            System.IO.File.Move(msiFilePath, newMSIpath);

            #endregion




            //Console.Read();
        }

        private string GetMsiVersionProperty(string msiFilePath, string property)
        {
            string retVal = string.Empty;

            // Create an Installer instance  
            WindowsInstaller.Installer installer = (WindowsInstaller.Installer) new Installer();

            // Open the msi file for reading  
            // 0 - Read, 1 - Read/Write  
            Database db = installer.OpenDatabase(msiFilePath, WindowsInstaller.MsiOpenDatabaseMode.msiOpenDatabaseModeReadOnly); //// Open the MSI database in the input file 

            // Fetch the requested property  
            string sql = String.Format(
                "SELECT Value FROM Property WHERE Property='{0}'", property);
            View view = db.OpenView(sql);
            //View vw = db.OpenView(@"SELECT `Value` FROM `Property` WHERE `Property` = 'ProductVersion'");
            view.Execute(null);

            // Read in the fetched record  
            Record record = view.Fetch();
            if (record != null)
            {
                retVal = record.get_StringData(1);
                System.Runtime.InteropServices.Marshal.FinalReleaseComObject(record);
            }
            view.Close();

            System.Runtime.InteropServices.Marshal.FinalReleaseComObject(view);
            System.Runtime.InteropServices.Marshal.FinalReleaseComObject(db);

            return retVal;
        }

    }
}