Pregunta Generar archivos manifiestos para COM sin registro


Tengo algunas aplicaciones (algunas nativas, algunas .NET) que usan archivos de manifiesto para que puedan ser desplegado en completo aislamiento, sin requerir ningún registro COM global. Por ejemplo, la dependencia del servidor com dbgrid32.ocx se declara de la siguiente manera en el archivo myapp.exe.manifest que se encuentra en la misma carpeta que myapp.exe:

<?xml version="1.0" encoding="utf-8" standalone="yes"?>
<assembly manifestVersion="1.0" xmlns="urn:schemas-microsoft-com:asm.v1">
  <assemblyIdentity type="win32" name="myapp.exe" version="1.2.3.4" />
  <dependency>
    <dependentAssembly>
      <assemblyIdentity type="win32" name="dbgrid32.ocx" version="5.1.81.4" />
    </dependentAssembly>
  </dependency>
</assembly>

El dbgrid32.ocx se implementa en la misma carpeta, junto con su propio archivo dbgrid32.ocx.manifest:

<?xml version="1.0" encoding="utf-8" standalone="yes"?>
<assembly manifestVersion="1.0" xmlns="urn:schemas-microsoft-com:asm.v1">
  <assemblyIdentity type="win32" name="dbgrid32.ocx" version="5.1.81.4" />
  <file name="dbgrid32.ocx">
     <typelib
        tlbid="{00028C01-0000-0000-0000-000000000046}"
        version="1.0"
        helpdir=""/>
    <comClass progid="MSDBGrid.DBGrid"
       clsid="{00028C00-0000-0000-0000-000000000046}"
       description="DBGrid  Control" />
  </file>
</assembly>

Todo esto funciona bien, pero mantener estos archivos de manifiesto manualmente es un poco molesto. ¿Hay alguna manera de generar estos archivos automáticamente? Idealmente, me gustaría declarar la dependencia de la aplicación en una lista de servidores COM (nativos y .NET) y luego permitir que el resto se genere automáticamente. ¿Es posible?


76
2018-01-21 16:07


origen


Respuestas:


Parece que la solución perfecta aún no existe. Para resumir algunas investigaciones:

Hacer mi manifiesto (enlazar)

Esta herramienta escanea un proyecto VB6 para buscar dependencias COM, pero también admite la declaración manual de las dependencias COM de última hora (es decir, las utilizadas a través de CreateObject).

Curiosamente, esta herramienta pone toda la información sobre las dependencias dentro del manifiesto de la aplicación. El exe de la aplicación y sus dependencias se describen como un único conjunto que consta de varios archivos. No me había dado cuenta antes de que esto fuera posible.

Parece una herramienta muy buena, pero a partir de la versión 0.6.6 tiene las siguientes limitaciones:

  • solo para aplicaciones VB6, comienza del archivo de proyecto VB6. Es una pena, porque Mucho de lo que hace realmente no tiene nada que ver con VB6.
  • aplicación de estilo asistente, no Adecuado para integrarse en una construcción proceso. Esto no es un gran problema si tu las dependencias no cambian mucho.
  • freeware sin fuente, arriesgado confiar en él porque podría convertirse en abandware en cualquier momento.

No probé si es compatible con las bibliotecas .NET com.

regsvr42 (enlace del proyecto de código)

Esta herramienta de línea de comandos genera archivos de manifiesto para bibliotecas COM nativas. Invoca a DllRegisterServer y luego espía el auto registro mientras agrega información en el registro. También puede generar un manifiesto de cliente para aplicaciones.

Esta utilidad no es compatible con las bibliotecas COM de .NET, ya que estas no exponen una rutina DllRegisterServer.

La utilidad está escrita en C ++. El código fuente está disponible.

mt.exe

Parte del SDK de Windows (se puede descargar de MSDN), que ya tienes si tienes Visual Studio instalado. Es documentado aquí. Puede generar archivos de manifiesto para bibliotecas COM nativas con este modo:

mt.exe -tlb:mycomlib.ocx -dll:mycomlib.ocx -out:mycomlib.ocx.manifest

Puede generar archivos de manifiesto para las bibliotecas COM de .NET con este modo:

mt.exe -managedassemblyname:netlib.dll -nodependency -out:netlib.dll.manifest

Sin embargo, hay algunos problemas con esta herramienta:

  • El primer fragmento no generará atributos progid, rompiendo clientes que usan CreateObject con los progids.
  • El segundo fragmento generará <runtime>y <mvid> elementos que necesitan ser eliminados antes los manifiestos realmente funcionan
  • Generación de clientes se manifiesta para aplicaciones no es compatible.

Quizás las futuras versiones de SDK mejorarán esta herramienta, probé la de Windows SDK 6.0a (vista).


56
2018-01-25 00:50



Con la tarea MSBuild GenerateApplicationManifest Genere un manifiesto en la línea de comando idéntico al manifiesto que genera Visual Studio. Sospecho que Visual Studio usa el GenerateApplicationManifest durante la construcción. A continuación se muestra mi script de compilación que se puede ejecutar desde la línea de comandos utilizando msbuild "msbuild build.xml"

Gracias a Dave Templin y su publicación que me indicó la tarea GenerateApplicationManifesty MSDN's documentación adicional de la tarea.

build.xml

<?xml version="1.0" encoding="utf-8"?>
<Project DefaultTargets="Build" xmlns="http://schemas.microsoft.com/developer/msbuild/2003">
    <Target Name="Build">
        <ItemGroup>
            <File Include='MyNativeApp.exe'/>
            <ComComponent Include='Com1.ocx;Com2.ocx'/>
        </ItemGroup>
        <GenerateApplicationManifest
            AssemblyName="MyNativeApp.exe"
            AssemblyVersion="1.0.0.0"
            IsolatedComReferences="@(ComComponent)"
            Platform="x86"
            ManifestType="Native">
            <Output
                ItemName="ApplicationManifest"
                TaskParameter="OutputManifest"/>
        </GenerateApplicationManifest>
    </Target>   
</Project>

23
2018-06-14 15:22



Make My Manifest (MMM) es una buena herramienta para hacer esto. También es posible escribir un script para procesar todos sus archivos DLL / OCX usando mt.exe generar un manifiesto para cada uno y luego fusionarlos todos juntos. MMM es generalmente mejor / más fácil, porque también maneja muchos casos especiales / extraños.


9
2018-01-21 16:17



Puedes usar Desatendido Make My Manifest spin off para generar manifiestos directamente en construcciones automatizadas. Utiliza un archivo de script para agregar componentes COM dependientes. Este es un extracto de la muestra ini con los comandos disponibles:

# Unattended MMM script
#
# Command names are case-insensitive. Reference of supported commands:
#
# Command: Identity
#
#   Appends assemblyIdentity and description tags.
#
#   Parameters       <exe_file> [name] [description]
#      exe_file      file name can be quoted if containing spaces. The containing folder 
#                    of the executable sets base path for relative file names
#      name          (optional) assembly name. Defaults to MyAssembly
#      description   (optional) description of assembly
#
# Command: Dependency
#
#   Appends dependency tag for referencing dependent assemblies like Common Controls 6.0, 
#     VC run-time or MFC
#
#   Parameters       {<lib_name>|<assembly_file>} [version] [/update]
#     lib_name       one of { comctl, vc90crt, vc90mfc }
#     assembly_file  file name of .NET DLL exporting COM classes
#     version        (optional) required assembly version. Multiple version of vc90crt can
#                    be required by a single manifest
#     /update        (optional) updates assembly_file assembly manifest. Spawns mt.exe
#
# Command: File
#
#   Appends file tag and collects information about coclasses and interfaces exposed by 
#     the referenced COM component typelib.
#
#   Parameters       <file_name> [interfaces]
#     file_name      file containing typelib. Can be relative to base path
#     interfaces     (optional) pipe (|) separated interfaces with or w/o leading 
#                    underscore
#
# Command: Interface
#
#   Appends comInterfaceExternalProxyStub tag for inter-thread marshaling of interfaces
#
#   Parameters       <file_name> <interfaces>
#     file_name      file containing typelib. Can be relative to base path
#     interfaces     pipe (|) separated interfaces with or w/o leading underscore
#
# Command: TrustInfo
#
#   Appends trustInfo tag for UAC user-rights elevation on Vista and above
#
#   Parameters       [level] [uiaccess]
#     level          (optional) one of { 1, 2, 3 } corresponding to { asInvoker, 
#                    highestAvailable, requireAdministrator }. Default is 1
#     uiaccess       (optional) true/false or 0/1. Allows application to gain access to 
#                    the protected system UI. Default is 0
#
# Command: DpiAware
#
#   Appends dpiAware tag for custom DPI aware applications
#
#   Parameters       [on_off]
#     on_off         (optional) true/false or 0/1. Default is 0
#
# Command: SupportedOS
#
#   Appends supportedOS tag
#
#   Parameters       <os_type>
#     os_type        one of { vista, win7 }. Multiple OSes can be supported by a single 
#                    manifest
#

7
2017-10-14 19:22



Para completar los ProgID que mt.exe no incluye, puede llamar ProgIDFromCLSID para buscarlos desde el registro. Esto requiere el registro COM tradicional antes de completar el archivo de manifiesto, pero posteriormente, el archivo de manifiesto será autosuficiente.

Este código C # agrega los ProgID a todas las clases COM en un manifiesto:

var manifest = XDocument.Load(fileName);
var namespaceManager = new XmlNamespaceManager(new NameTable());
namespaceManager.AddNamespace("s", "urn:schemas-microsoft-com:asm.v1");
foreach (var classElement in manifest.XPathSelectElements("s:assembly/s:file/s:comClass", namespaceManager)) {
    var clsid = Guid.Parse(classElement.Attribute("clsid").Value);
    int result = ProgIDFromCLSID(ref clsid, out string progId); if (result != S_OK) throw new COMException($"ProgID lookup failed for {clsid}.", result);
    classElement.SetAttributeValue("progid", progId);
}
manifest.Save(fileName);

El código se basa en estas definiciones de interoperabilidad:

[DllImport("ole32.dll")] static extern int ProgIDFromCLSID([In] ref Guid clsid, [MarshalAs(UnmanagedType.LPWStr)] out string lplpszProgID);
const int S_OK = 0;

0
2018-04-06 02:32