forked from dotnet/sdk
-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy pathResolveToolPackagePaths.cs
124 lines (100 loc) · 4.16 KB
/
ResolveToolPackagePaths.cs
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
// Copyright (c) .NET Foundation and contributors. All rights reserved.
// Licensed under the MIT license. See LICENSE file in the project root for full license information.
using System;
using System.Collections.Generic;
using System.IO;
using System.Linq;
using System.Runtime.Versioning;
using System.Text;
using Microsoft.Build.Framework;
using Microsoft.Build.Utilities;
using NuGet.Frameworks;
namespace Microsoft.NET.Build.Tasks
{
/// <summary>
/// 1. Publish PublishRelativePath is a relative <strong>file</strong> path. This need to convert to relative path.
/// 2. Due to DotnetTools package format, "tools/{TargetFramework}/any/" is needed in addition to the relative path.
/// </summary>
public sealed class ResolveToolPackagePaths : TaskBase
{
public string AppHostIntermediatePath { get; set; }
[Required]
public ITaskItem[] ResolvedFileToPublish { get; set; }
[Required]
public string PublishDir { get; set; }
[Required]
public string TargetFrameworkMoniker { get; set; }
[Output]
public ITaskItem[] ResolvedFileToPublishWithPackagePath { get; private set; }
private const char ForwardSeparatorChar = '\\';
private const char BackwardSeparatorChar = '/';
protected override void ExecuteCore()
{
var result = new List<TaskItem>();
foreach (ITaskItem r in ResolvedFileToPublish)
{
// skip packing apphost since a dotnet tool will get a generated shim executable once installed
if (r.ItemSpec.Equals(AppHostIntermediatePath, StringComparison.Ordinal))
{
continue;
}
string relativePath = r.GetMetadata("RelativePath");
var fullpath = Path.GetFullPath(
Path.Combine(PublishDir,
relativePath));
var i = new TaskItem(fullpath);
var shortFrameworkName = NuGetFramework.Parse(TargetFrameworkMoniker).GetShortFolderName();
i.SetMetadata("PackagePath", $"tools/{shortFrameworkName}/any/{GetDirectoryPathInRelativePath(relativePath)}");
result.Add(i);
}
ResolvedFileToPublishWithPackagePath = result.ToArray();
}
/// <summary>
/// Change "dir/file.exe" to "dir"
/// </summary>
internal static string GetDirectoryPathInRelativePath(string publishRelativePath)
{
publishRelativePath = NormalizeDirectorySeparatorsToUnixStyle(publishRelativePath);
var index = publishRelativePath.LastIndexOf(BackwardSeparatorChar);
return index == -1 ? string.Empty : publishRelativePath.Substring(0, index);
}
// https://github.com/dotnet/corefx/issues/4208
// Basic copy paste from corefx. But Normalize to "/" instead of \
private static string NormalizeDirectorySeparatorsToUnixStyle(string path)
{
if (string.IsNullOrEmpty(path))
{
return path;
}
char current;
StringBuilder builder = new StringBuilder(path.Length);
int start = 0;
if (IsDirectorySeparator(path[start]))
{
start++;
builder.Append(BackwardSeparatorChar);
}
for (int i = start; i < path.Length; i++)
{
current = path[i];
// If we have a separator
if (IsDirectorySeparator(current))
{
// If the next is a separator, skip adding this
if (i + 1 < path.Length && IsDirectorySeparator(path[i + 1]))
{
continue;
}
// Ensure it is the primary separator
current = BackwardSeparatorChar;
}
builder.Append(current);
}
return builder.ToString();
}
private static bool IsDirectorySeparator(char c)
{
return c == ForwardSeparatorChar || c == BackwardSeparatorChar;
}
}
}