如何在我的引导程序 exe 中嵌入 MSI

How to embed an MSI in my bootstrapper exe

我为我的应用程序编写了一个 C++ 安装程序 bootstrap(硬性要求,因为我需要完全控制文件的格式)。现在我正试图弄清楚如何以最少的开销将我的 MSI 嵌入到我的 bootstrap EXE 中。我所有的搜索都向我展示了相反的情况(如何在 MSI 中嵌入 EXE)。

此安装程序必须离线工作,所以我 bootstrap 无法下载文件。

我当然可以将它作为字节流嵌入到我的源文件中,但这会导致文件大小的大量开销。

将数据作为资源嵌入到二进制文件中并提取它是相当常见的。这种事:

http://www.codeproject.com/Articles/528178/Load-DLL-From-Embedded-Resource

http://www.codeproject.com/Articles/4221/Adding-and-extracting-binary-resources

IIRC,编译器会在构建时通过适当的调整嵌入数据,所以你只需要提取它。

下面是如何提取数据的示例。

#include "stdafx.h"
#include "resource.h"
#include <iostream>
#include <fstream>
#include <iterator>
#include <vector>
#include <Windows.h>
#include <shellapi.h>

using namespace std;

void extract_bin_resource(std::wstring strCustomResName, int nResourceId, std::string strOutputPath)
{
    HGLOBAL hResourceLoaded;   // handle to loaded resource
    HRSRC   hRes;              // handle/ptr to res. info.
    char    *lpResLock;        // pointer to resource data
    DWORD   dwSizeRes;
    std::string strOutputLocation;
    std::string strAppLocation;

    hRes = FindResource(NULL, MAKEINTRESOURCE(nResourceId), strCustomResName.c_str());

    hResourceLoaded = LoadResource(NULL, hRes);
    lpResLock = (char *)LockResource(hResourceLoaded);
    dwSizeRes = SizeofResource(NULL, hRes);

    std::ofstream outputFile(strOutputPath.c_str(), std::ios::binary);
    outputFile.write((const char *)lpResLock, dwSizeRes);
    outputFile.close();
}

int CALLBACK WinMain(HINSTANCE hInstance, HINSTANCE hPrevInstance, LPSTR lpCmdLine, int nCmdShow) {
    // Extract the bundled installer
    extract_bin_resource(L"MSI", IDR_MSI1, "installer.msi");

    // Execute the installer
    SHELLEXECUTEINFO ShExecInfo = { 0 };
    ShExecInfo.cbSize = sizeof(SHELLEXECUTEINFO);
    ShExecInfo.fMask = SEE_MASK_NOCLOSEPROCESS;
    ShExecInfo.hwnd = NULL;
    ShExecInfo.lpVerb = NULL;
    ShExecInfo.lpFile = L"msiexec.exe";
    ShExecInfo.lpParameters = L"/i installer.msi";
    ShExecInfo.lpDirectory = NULL;
    ShExecInfo.nShow = SW_SHOW;
    ShExecInfo.hInstApp = NULL;
    ShellExecuteEx(&ShExecInfo);
    WaitForSingleObject(ShExecInfo.hProcess, INFINITE);

    // Delete the installer
    DeleteFile(L"installer.msi");

    return 0;
}