VS2008セットアッププロジェクトのoutput.msiファイル名の一部としてバージョン番号を含める方法はありますか?
たとえば、「myinstaller-1.0.13.msi」という出力ファイルが必要です。このファイルでは、展開プロジェクトのプロパティに入力したバージョン番号に基づいてバージョン部分が自動的に設定されます。
まだこれが必要かどうかはわかりませんが、ビルド後のイベントで同様の操作を行ったため、これに答えたいと思いました。私が行った調査では、セットアッププロセスを通じて内部的にファイル名を希望どおりに設定することはできません。
ビルド後のイベントで外部アプリケーションを介して出力ファイルに名前を付けることにより、他の方法でこれを行うことができます。
これがあなたができることです:
ビルド後のイベント->
[MsiRenamerAppPath]\MsiRenamer.exe "$(BuildOutputPath)"
Msiファイルの名前をデプロイメントプロジェクトのバージョン番号に変更するアプリケーションを作成します。以下は、アプリケーションに使用されるコードです。これは私が推測するあなたの要件を満たすはずです。
Msiプロパティコードの取得は 代替記事 から使用されます
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;
}
}
上記の.exeメソッドを使用したくなかったので、少し時間に余裕があったので、じっくり考え始めました。 Windows 764ビットでVS2008を使用しています。セットアッププロジェクトがある場合、それをMySetupと呼びましょう。プロジェクトのすべての詳細は、ファイル$(ProjectDir)MySetup.vdprojにあります。
製品バージョンは、そのファイルの1行にフォームで表示されます。
ProductVersion="8:1.0.0"
ここで、セットアッププロジェクトにISビルド後のイベントがあります。セットアッププロジェクトを選択してF4キーを押すと、右クリックしてプロパティを選択した場合とはまったく異なるプロパティのセットが表示されます。 F4キーを押すと、そのうちの1つが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が含まれている残りの1行を返し、それをファイルに出力します。次に、引用符、スペースを削除し、ドットをアンダースコアに変換します。
For行は、コロンの残りの文字列を分割し、2番目の部分を取得して、ファイルに再度出力します。
次に、realvarをファイルに残っている値に設定し、MySetup.msiの名前を変更して日付とバージョンを含めます。
したがって、上記のProductVersionを考えると、2012年3月27日の場合、ファイルの名前は次のように変更されます。
MySetup-20120327-1_0_0.msi
明らかにこの方法を使用すると、vdprojファイル内の任意の変数を取得して出力ファイル名に含めることができ、それを行うために追加の.exeプログラムを構築する必要はありません。
HTH
Jim Grimmettの回答と同じ概念ですが、依存関係が少なくなっています。
_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 ...
_
(VS Setup&Deploymentプロジェクトではなく)WIXプロジェクトを使用する場合、 この記事 は、目的を達成する方法を正確に説明しています。
私はPowerShellで2行でそれを行いました。
$versionText=(Get-Item MyProgram.exe).VersionInfo.FileVersion
(Get-Content MySetup.vdproj.template).replace('${VERSION}', $($versionText)) | Set-Content MySetup.vdproj
既存の.vdprojの名前をMySetup.vdproj.templateに変更し、プライマリexeファイルのバージョンを挿入する場所に「$ {VERSION}」を挿入します。
次に、VSはvdprojファイルの変更を検出し、それをリロードするかどうかを尋ねます。
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's 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;
}
}
}