用于创建Visual Studio项目模板扩展Zip问题的PowerShell脚本

时间:2016-11-07 10:09:59

标签: powershell visual-studio-extensions

我尝试编写PowerShell脚本来编写Visual Studio扩展,只需添加项目模板即可。这是一个用于演示问题的脚本的精简版本:

# Add the assemblies

Add-Type -Assembly System.IO.Compression.FileSystem

# Create temporary directories for the zip archives

[System.IO.Directory]::CreateDirectory("Extension")
[System.IO.Directory]::CreateDirectory("Template")

# Build up the contents of the template file

$templateContent = "<?xml version=`"1.0`" encoding=`"utf-8`"?>`r`n"
$templateContent += "<VSTemplate Version=`"3.0.0`" Type=`"Project`" xmlns=`"http://schemas.microsoft.com/developer/vstemplate/2005`" xmlns:sdk=`"http://schemas.microsoft.com/developer/vstemplate-sdkextension/2010`">`r`n"
$templateContent += "    <TemplateData>`r`n"
$templateContent += "        <Name>MyExtension</Name>`r`n"
$templateContent += "        <Description>MyExtension</Description>`r`n"
$templateContent += "        <Icon>MyExtension.ico</Icon>`r`n"
$templateContent += "        <ProjectType>CSharp</ProjectType>`r`n"
$templateContent += "        <ProjectSubType></ProjectSubType>`r`n"
$templateContent += "        <RequiredFrameworkVersion>2.0</RequiredFrameworkVersion>`r`n"
$templateContent += "        <SortOrder>1000</SortOrder>`r`n"
$templateContent += "        <TemplateID>61251892-9605-4816-846b-858352383c38</TemplateID>`r`n"
$templateContent += "        <CreateNewFolder>true</CreateNewFolder>`r`n"
$templateContent += "        <DefaultName>MyExtension</DefaultName>`r`n"
$templateContent += "        <ProvideDefaultName>true</ProvideDefaultName>`r`n"
$templateContent += "    </TemplateData>`r`n"
$templateContent += "    <TemplateContent>`r`n"
$templateContent += "        <Project File=`"MyExtension.csproj`" ReplaceParameters=`"true`"></Project>`r`n"
$templateContent += "    </TemplateContent>`r`n"
$templateContent += "</VSTemplate>"

# Save the template file

$templateContent | Out-File ([System.IO.Path]::Combine("Template", "MyExtension.vstemplate")) -Encoding "UTF8" -NoNewline

# Build up the contents of the proj file

$projContent = "<?xml version=`"1.0`" encoding=`"utf-8`"?>`r`n"
$projContent += "<Project ToolsVersion=`"4.0`" DefaultTargets=`"Build`" xmlns=`"http://schemas.microsoft.com/developer/msbuild/2003`">`r`n"
$projContent += "  <Import Project=`"`$(MSBuildExtensionsPath)\`$(MSBuildToolsVersion)\Microsoft.Common.props`" Condition=`"Exists('`$(MSBuildExtensionsPath)\`$(MSBuildToolsVersion)\Microsoft.Common.props')`" />`r`n"
$projContent += "  <PropertyGroup>`r`n"
$projContent += "    <Configuration Condition=`" '`$(Configuration)' == '' `">Debug</Configuration>`r`n"
$projContent += "    <Platform Condition=`" '`$(Platform)' == '' `">AnyCPU</Platform>`r`n"
$projContent += "    <ProductVersion>`r`n"
$projContent += "    </ProductVersion>`r`n"
$projContent += "    <SchemaVersion>2.0</SchemaVersion>`r`n"
$projContent += "    <ProjectGuid>{403C08FA-9E44-4A8A-A757-1662142E1334}</ProjectGuid>`r`n"
$projContent += "    <ProjectTypeGuids>{349c5851-65df-11da-9384-00065b846f21};{fae04ec0-301f-11d3-bf4b-00c04f79efbc}</ProjectTypeGuids>`r`n"
$projContent += "    <OutputType>Library</OutputType>`r`n"
$projContent += "    <AppDesignerFolder>Properties</AppDesignerFolder>`r`n"
$projContent += "    <RootNamespace>`$safeprojectname`$</RootNamespace>`r`n"
$projContent += "    <AssemblyName>`$safeprojectname`$</AssemblyName>`r`n"
$projContent += "    <TargetFrameworkVersion>v4.5</TargetFrameworkVersion>`r`n"
$projContent += "    <UseIISExpress>false</UseIISExpress>`r`n"
$projContent += "    <IISExpressSSLPort />`r`n"
$projContent += "    <IISExpressAnonymousAuthentication />`r`n"
$projContent += "    <IISExpressWindowsAuthentication />`r`n"
$projContent += "    <IISExpressUseClassicPipelineMode />`r`n"
$projContent += "  </PropertyGroup>`r`n"
$projContent += "  <PropertyGroup Condition=`" '`$(Configuration)|`$(Platform)' == 'Debug|AnyCPU' `">`r`n"
$projContent += "    <DebugSymbols>true</DebugSymbols>`r`n"
$projContent += "    <DebugType>full</DebugType>`r`n"
$projContent += "    <Optimize>false</Optimize>`r`n"
$projContent += "    <OutputPath>bin\</OutputPath>`r`n"
$projContent += "    <DefineConstants>DEBUG;TRACE</DefineConstants>`r`n"
$projContent += "    <ErrorReport>prompt</ErrorReport>`r`n"
$projContent += "    <WarningLevel>4</WarningLevel>`r`n"
$projContent += "  </PropertyGroup>`r`n"
$projContent += "  <PropertyGroup Condition=`" '`$(Configuration)|`$(Platform)' == 'Release|AnyCPU' `">`r`n"
$projContent += "    <DebugType>pdbonly</DebugType>`r`n"
$projContent += "    <Optimize>true</Optimize>`r`n"
$projContent += "    <OutputPath>bin\</OutputPath>`r`n"
$projContent += "    <DefineConstants>TRACE</DefineConstants>`r`n"
$projContent += "    <ErrorReport>prompt</ErrorReport>`r`n"
$projContent += "    <WarningLevel>4</WarningLevel>`r`n"
$projContent += "  </PropertyGroup>`r`n"
$projContent += "  <ItemGroup>`r`n"
$projContent += "    <Reference Include=`"Microsoft.CSharp`" />`r`n"
$projContent += "    <Reference Include=`"System.ServiceModel`" />`r`n"
$projContent += "    <Reference Include=`"System.Transactions`" />`r`n"
$projContent += "    <Reference Include=`"System.Web.DynamicData`" />`r`n"
$projContent += "    <Reference Include=`"System.Web.Entity`" />`r`n"
$projContent += "    <Reference Include=`"System.Web.ApplicationServices`" />`r`n"
$projContent += "    <Reference Include=`"System.ComponentModel.DataAnnotations`" />`r`n"
$projContent += "    <Reference Include=`"System`" />`r`n"
$projContent += "    <Reference Include=`"System.Data`" />`r`n"
$projContent += "    <Reference Include=`"System.Core`" />`r`n"
$projContent += "    <Reference Include=`"System.Data.DataSetExtensions`" />`r`n"
$projContent += "    <Reference Include=`"System.Web.Extensions`" />`r`n"
$projContent += "    <Reference Include=`"System.Xml.Linq`" />`r`n"
$projContent += "    <Reference Include=`"System.Drawing`" />`r`n"
$projContent += "    <Reference Include=`"System.Web`" />`r`n"
$projContent += "    <Reference Include=`"System.Xml`" />`r`n"
$projContent += "    <Reference Include=`"System.Configuration`" />`r`n"
$projContent += "    <Reference Include=`"System.Web.Services`" />`r`n"
$projContent += "    <Reference Include=`"System.EnterpriseServices`" />`r`n"
$projContent += "  </ItemGroup>`r`n"
$projContent += "  <PropertyGroup>`r`n"
$projContent += "    <VisualStudioVersion Condition=`"'`$(VisualStudioVersion)' == ''`">10.0</VisualStudioVersion>`r`n"
$projContent += "    <VSToolsPath Condition=`"'`$(VSToolsPath)' == ''`">`$(MSBuildExtensionsPath32)\Microsoft\VisualStudio\v`$(VisualStudioVersion)</VSToolsPath>`r`n"
$projContent += "  </PropertyGroup>`r`n"
$projContent += "  <Import Project=`"`$(MSBuildBinPath)\Microsoft.CSharp.targets`" />`r`n"
$projContent += "  <Import Project=`"`$(VSToolsPath)\WebApplications\Microsoft.WebApplication.targets`" Condition=`"'`$(VSToolsPath)' != ''`" />`r`n"
$projContent += "  <Import Project=`"`$(MSBuildExtensionsPath32)\Microsoft\VisualStudio\v10.0\WebApplications\Microsoft.WebApplication.targets`" Condition=`"false`" />`r`n"
$projContent += "  <ProjectExtensions>`r`n"
$projContent += "    <VisualStudio>`r`n"
$projContent += "      <FlavorProperties GUID=`"{349c5851-65df-11da-9384-00065b846f21}`">`r`n"
$projContent += "        <WebProjectProperties>`r`n"
$projContent += "          <UseIIS>False</UseIIS>`r`n"
$projContent += "          <AutoAssignPort>True</AutoAssignPort>`r`n"
$projContent += "          <DevelopmentServerPort>58060</DevelopmentServerPort>`r`n"
$projContent += "          <DevelopmentServerVPath>/</DevelopmentServerVPath>`r`n"
$projContent += "          <IISUrl>`r`n"
$projContent += "          </IISUrl>`r`n"
$projContent += "          <NTLMAuthentication>False</NTLMAuthentication>`r`n"
$projContent += "          <UseCustomServer>True</UseCustomServer>`r`n"
$projContent += "          <CustomServerUrl>http://localhost/</CustomServerUrl>`r`n"
$projContent += "          <SaveServerSettingsInUserFile>False</SaveServerSettingsInUserFile>`r`n"
$projContent += "        </WebProjectProperties>`r`n"
$projContent += "      </FlavorProperties>`r`n"
$projContent += "    </VisualStudio>`r`n"
$projContent += "  </ProjectExtensions>`r`n"
$projContent += "  <!-- To modify your build process, add your task inside one of the targets below and uncomment it. `r`n"
$projContent += "       Other similar extension points exist, see Microsoft.Common.targets.`r`n"
$projContent += "  <Target Name=`"BeforeBuild`">`r`n"
$projContent += "  </Target>`r`n"
$projContent += "  <Target Name=`"AfterBuild`">`r`n"
$projContent += "  </Target>`r`n"
$projContent += "  -->`r`n"
$projContent += "</Project>"

# Save the proj file

$projContent | Out-File ([System.IO.Path]::Combine("Template", "MyExtension.csproj")) -Encoding "UTF8" -NoNewline

# Create the template zip file

[System.IO.Directory]::CreateDirectory("Extension\ProjectTemplates\CSharp\Web\1033")
[System.IO.Compression.ZipFile]::CreateFromDirectory("Template", "Extension\ProjectTemplates\CSharp\Web\1033\MyExtension.zip")

# Create a content types xml file (an error will be thrown if this does not exist)

$conentTypesContent = "<?xml version=`"1.0`" encoding=`"utf-8`"?><Types xmlns=`"http://schemas.openxmlformats.org/package/2006/content-types`"><Default Extension=`"vsixmanifest`" ContentType=`"text/xml`" /><Default Extension=`"zip`" ContentType=`"application/zip`" /></Types>"

# Save the content types file

$conentTypesContent | Out-File -literalPath "Extension\[Content_Types].xml" -Encoding "UTF8" -NoNewline

# Now create an extension manifest for the visual studio template

$extensionContent = "<PackageManifest Version=`"2.0.0`" xmlns=`"http://schemas.microsoft.com/developer/vsx-schema/2011`">`r`n"
$extensionContent += "    <Metadata>`r`n"
$extensionContent += "        <Identity Id=`"MyExtension - 1`" Version=`"0.1.0`" Language=`"en-US`" Publisher=`"MyExtension.net Ltd`" />`r`n"
$extensionContent += "        <DisplayName>MyExtension Project Template</DisplayName>`r`n"
$extensionContent += "        <Description xml:space=`"preserve`">MyExtension Project Template Extension</Description>`r`n"
$extensionContent += "    </Metadata>`r`n"
$extensionContent += "    <Installation>`r`n"
$extensionContent += "        <InstallationTarget Id=`"Microsoft.VisualStudio.Community`" Version=`"[14.0]`" />`r`n"
$extensionContent += "    </Installation>`r`n"
$extensionContent += "    <Dependencies>`r`n"
$extensionContent += "        <Dependency Id=`"Microsoft.Framework.NDP`" DisplayName=`"Microsoft .NET Framework`" Version=`"[4.5,)`" />`r`n"
$extensionContent += "    </Dependencies>`r`n"
$extensionContent += "    <Assets>`r`n"
$extensionContent += "        <Asset Type=`"Microsoft.VisualStudio.ProjectTemplate`" Path=`"ProjectTemplates`" />`r`n"
$extensionContent += "    </Assets>`r`n"
$extensionContent += "</PackageManifest>"

# Save the extension file

$extensionContent | Out-File "Extension\extension.vsixmanifest" -Encoding "UTF8" -NoNewline

# Create the extension zip file

[System.IO.Compression.ZipFile]::CreateFromDirectory("Extension", "MyExtension.vsix")

# Delete the temporary directories
[System.IO.Directory]::Delete("Extension", $true)
[System.IO.Directory]::Delete("Template", $true)

当我运行脚本时,它成功生成了一个MyExtension.vsix文件。当我执行该文件时,它会安装扩展,但如果我现在打开Visual Studio 2015并创建一个新项目,则项目模板(C# - &gt; Web)不存在。

但是,如果您执行以下步骤,那么它可以正常工作(请确保您先卸载旧的扩展名):

  1. 将MyExtension.vsix文件重命名为MyExtension.zip,然后解压缩到目录
  2. 进入该目录并选择所有文件,右键单击并选择发送到压缩文件夹
  3. 将生成的zip文件重命名为MyExtension.vsix
  4. 执行此文件以安装扩展程序
  5. 我花了好几个小时甚至注意到这种情况解决了这个问题。然而,这是一个黑客攻击,从长远来看,它将变得相当乏味。我想知道是否有人知道我的PowerShell脚本如何更改以解决此问题。

    由于

1 个答案:

答案 0 :(得分:3)

由于某种原因,[System.IO.Compression.ZipFile] :: CreateFromDirectory不会创建一个正常工作的zip / vsix文件,     即使它会显示为已安装。模板未显示在新项目UI中。

使用7zip代替创建zip文件。

虽然我试图调查此问题,但我不喜欢代码没有使用完全限定的路径,并且字符串很难查看。我稍微重构了你的代码。

根据我的测试,现在按预期工作。

<强> CODE

<#
http://stackoverflow.com/questions/40462544/powershell-script-to-create-visual-studio-project-template-extension-zip-issue

For some reason, [System.IO.Compression.ZipFile]::CreateFromDirectory will not create a zip/vsix file that works correctly,
even though it will show as installed.  The template does not show in the new project UI.

Use 7zip instead to create zip files.
#>

Set-StrictMode -Version Latest
$VerbosePreference = [System.Management.Automation.ActionPreference]::Continue

# Makes debugging from ISE easier.
if ($PSScriptRoot -eq "")
{
    $root = Split-Path -Parent $psISE.CurrentFile.FullPath
}
else
{
    $root = $PSScriptRoot
}

Set-Location $root

<#
Create a zip file with items under Path in the root of the zip file.
#>
function New-ZipFile([string]$Path, [string]$FileName)
{
    $zipExe = 'C:\Program Files\7-Zip\7z.exe'
    $currentLocation = Get-Location
    Set-Location $Path
    & $zipExe a -tzip $FileName * -r
    Set-Location $currentLocation
}

# Create temporary directories for the zip archives

"Extension", "Template" | % {New-Item (Join-Path $root $_) -ItemType Directory}

# Build up the contents of the template file

$templateContent = @'
<?xml version="1.0" encoding="utf-8"?>
<VSTemplate Version="3.0.0" Type="Project" xmlns="http://schemas.microsoft.com/developer/vstemplate/2005" xmlns:sdk="http://schemas.microsoft.com/developer/vstemplate-sdkextension/2010">
    <TemplateData>
        <Name>MyExtension</Name>
        <Description>MyExtension</Description>
        <Icon>MyExtension.ico</Icon>
        <ProjectType>CSharp</ProjectType>
        <ProjectSubType></ProjectSubType>
        <RequiredFrameworkVersion>2.0</RequiredFrameworkVersion>
        <SortOrder>1000</SortOrder>
        <TemplateID>61251892-9605-4816-846b-858352383c38</TemplateID>
        <CreateNewFolder>true</CreateNewFolder>
        <DefaultName>MyExtension</DefaultName>
        <ProvideDefaultName>true</ProvideDefaultName>
    </TemplateData>
    <TemplateContent>
        <Project File="MyExtension.csproj" ReplaceParameters="true"></Project>
    </TemplateContent>
</VSTemplate>
'@

# Save the template file

$templateContent | Out-File (Join-Path $root "Template\MyExtension.vstemplate") -Encoding "UTF8" #-NoNewline

# Build up the contents of the proj file

$projContent = @'
<?xml version="1.0" encoding="utf-8"?>
<Project ToolsVersion="4.0" DefaultTargets="Build" xmlns="http://schemas.microsoft.com/developer/msbuild/2003">
    <Import Project="$(MSBuildExtensionsPath)\$(MSBuildToolsVersion)\Microsoft.Common.props" Condition="Exists('$(MSBuildExtensionsPath)\$(MSBuildToolsVersion)\Microsoft.Common.props')" />
    <PropertyGroup>
    <Configuration Condition=" '$(Configuration)' == '' ">Debug</Configuration>
    <Platform Condition=" '$(Platform)' == '' ">AnyCPU</Platform>
    <ProductVersion>
    </ProductVersion>
    <SchemaVersion>2.0</SchemaVersion>
    <ProjectGuid>{403C08FA-9E44-4A8A-A757-1662142E1334}</ProjectGuid>
    <ProjectTypeGuids>{349c5851-65df-11da-9384-00065b846f21};{fae04ec0-301f-11d3-bf4b-00c04f79efbc}</ProjectTypeGuids>
    <OutputType>Library</OutputType>
    <AppDesignerFolder>Properties</AppDesignerFolder>
    <RootNamespace>$safeprojectname$</RootNamespace>
    <AssemblyName>$safeprojectname$</AssemblyName>
    <TargetFrameworkVersion>v4.5</TargetFrameworkVersion>
    <UseIISExpress>false</UseIISExpress>
    <IISExpressSSLPort />
    <IISExpressAnonymousAuthentication />
    <IISExpressWindowsAuthentication />
    <IISExpressUseClassicPipelineMode />
    </PropertyGroup>
    <PropertyGroup Condition=" '$(Configuration)|$(Platform)' == 'Debug|AnyCPU' ">
    <DebugSymbols>true</DebugSymbols>
    <DebugType>full</DebugType>
    <Optimize>false</Optimize>
    <OutputPath>bin\</OutputPath>
    <DefineConstants>DEBUG;TRACE</DefineConstants>
    <ErrorReport>prompt</ErrorReport>
    <WarningLevel>4</WarningLevel>
    </PropertyGroup>
    <PropertyGroup Condition=" '$(Configuration)|$(Platform)' == 'Release|AnyCPU' ">
    <DebugType>pdbonly</DebugType>
    <Optimize>true</Optimize>
    <OutputPath>bin\</OutputPath>
    <DefineConstants>TRACE</DefineConstants>
    <ErrorReport>prompt</ErrorReport>
    <WarningLevel>4</WarningLevel>
    </PropertyGroup>
    <ItemGroup>
    <Reference Include="Microsoft.CSharp" />
    <Reference Include="System.ServiceModel" />
    <Reference Include="System.Transactions" />
    <Reference Include="System.Web.DynamicData" />
    <Reference Include="System.Web.Entity" />
    <Reference Include="System.Web.ApplicationServices" />
    <Reference Include="System.ComponentModel.DataAnnotations" />
    <Reference Include="System" />
    <Reference Include="System.Data" />
    <Reference Include="System.Core" />
    <Reference Include="System.Data.DataSetExtensions" />
    <Reference Include="System.Web.Extensions" />
    <Reference Include="System.Xml.Linq" />
    <Reference Include="System.Drawing" />
    <Reference Include="System.Web" />
    <Reference Include="System.Xml" />
    <Reference Include="System.Configuration" />
    <Reference Include="System.Web.Services" />
    <Reference Include="System.EnterpriseServices" />
    </ItemGroup>
    <PropertyGroup>
    <VisualStudioVersion Condition="'$(VisualStudioVersion)' == ''">10.0</VisualStudioVersion>
    <VSToolsPath Condition="'$(VSToolsPath)' == ''">$(MSBuildExtensionsPath32)\Microsoft\VisualStudio\v$(VisualStudioVersion)</VSToolsPath>
    </PropertyGroup>
    <Import Project="$(MSBuildBinPath)\Microsoft.CSharp.targets" />
    <Import Project="$(VSToolsPath)\WebApplications\Microsoft.WebApplication.targets" Condition="'$(VSToolsPath)' != ''" />
    <Import Project="$(MSBuildExtensionsPath32)\Microsoft\VisualStudio\v10.0\WebApplications\Microsoft.WebApplication.targets" Condition="false" />
    <ProjectExtensions>
    <VisualStudio>
        <FlavorProperties GUID="{349c5851-65df-11da-9384-00065b846f21}">
        <WebProjectProperties>
            <UseIIS>False</UseIIS>
            <AutoAssignPort>True</AutoAssignPort>
            <DevelopmentServerPort>58060</DevelopmentServerPort>
            <DevelopmentServerVPath>/</DevelopmentServerVPath>
            <IISUrl>
            </IISUrl>
            <NTLMAuthentication>False</NTLMAuthentication>
            <UseCustomServer>True</UseCustomServer>
            <CustomServerUrl>http://localhost/</CustomServerUrl>
            <SaveServerSettingsInUserFile>False</SaveServerSettingsInUserFile>
        </WebProjectProperties>
        </FlavorProperties>
    </VisualStudio>
    </ProjectExtensions>
    <!-- To modify your build process, add your task inside one of the targets below and uncomment it. 
        Other similar extension points exist, see Microsoft.Common.targets.
    <Target Name="BeforeBuild">
    </Target>
    <Target Name="AfterBuild">
    </Target>
    -->
</Project>
'@

# Save the proj file

$projContent | Out-File (Join-Path $root "Template\MyExtension.csproj") -Encoding "UTF8" #-NoNewline

# Create the template zip file

New-Item (Join-Path $root "Extension\ProjectTemplates\CSharp\Web\1033") -ItemType Directory
New-ZipFile (Join-Path $root "Template") (Join-Path $root "Extension\ProjectTemplates\CSharp\Web\1033\MyExtension.zip")

# Create a content types xml file (an error will be thrown if this does not exist)

$conentTypesContent = @'
<?xml version="1.0" encoding="utf-8"?><Types xmlns="http://schemas.openxmlformats.org/package/2006/content-types"><Default Extension="vsixmanifest" ContentType="text/xml" /><Default Extension="zip" ContentType="application/zip" /></Types>
'@

# Save the content types file

$conentTypesContent | Out-File -literalPath (Join-Path $root "Extension\[Content_Types].xml") -Encoding "UTF8" #-NoNewline

# Now create an extension manifest for the visual studio template

$extensionContent = @'
<PackageManifest Version="2.0.0" xmlns="http://schemas.microsoft.com/developer/vsx-schema/2011">
    <Metadata>
        <Identity Id="MyExtension - 1" Version="0.1.0" Language="en-US" Publisher="MyExtension.net Ltd" />
        <DisplayName>MyExtension Project Template</DisplayName>
        <Description xml:space="preserve">MyExtension Project Template Extension</Description>
    </Metadata>
    <Installation>
        <InstallationTarget Id="Microsoft.VisualStudio.Community" Version="[14.0]" />
    </Installation>
    <Dependencies>
        <Dependency Id="Microsoft.Framework.NDP" DisplayName="Microsoft .NET Framework" Version="[4.5,)" />
    </Dependencies>
    <Assets>
        <Asset Type="Microsoft.VisualStudio.ProjectTemplate" Path="ProjectTemplates" />
    </Assets>
</PackageManifest>
'@

# Save the extension file

$extensionContent | Out-File (Join-Path $root "Extension\extension.vsixmanifest") -Encoding "UTF8" #-NoNewline

# Create the extension zip file

New-ZipFile (Join-Path $root "Extension") (Join-Path $root "MyExtension.vsix")

# Delete the temporary directories
"Extension", "Template" | % {Remove-Item (Join-Path $root $_) -Recurse -Force}