forked from dotnet/sourcelink
-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy pathGetSourceLinkUrl.cs
114 lines (95 loc) · 4.44 KB
/
GetSourceLinkUrl.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
// Copyright (c) Microsoft. All Rights Reserved. Licensed under the Apache License, Version 2.0. See License.txt in the project root for license information.
using System;
using System.Collections.Generic;
using System.IO;
using System.Linq;
using Microsoft.Build.Framework;
using Microsoft.Build.Tasks.SourceControl;
namespace Microsoft.SourceLink.AzureRepos.Git
{
public sealed class GetSourceLinkUrl : GetSourceLinkUrlGitTask
{
private const string UrlMapEnvironmentVariableName = "BUILD_REPOSITORY_URL_MAP";
protected override string HostsItemGroupName => "SourceLinkAzureReposGitHost";
protected override string ProviderDisplayName => "AzureRepos.Git";
protected override Uri GetDefaultContentUriFromHostUri(string authority, Uri gitUri)
{
var gitHost = gitUri.GetHost();
return AzureDevOpsUrlParser.IsVisualStudioHostedServer(gitHost) ?
new Uri($"{gitUri.Scheme}://{gitHost.Substring(0, gitHost.IndexOf('.'))}.{authority}", UriKind.Absolute) :
new Uri($"{gitUri.Scheme}://{authority}", UriKind.Absolute);
}
// Repository URL already contains account in case of VS host. Don't add it like we do when the content URL is inferred from host name.
protected override Uri GetDefaultContentUriFromRepositoryUri(Uri repositoryUri)
=> new Uri($"{repositoryUri.Scheme}://{repositoryUri.GetAuthority()}", UriKind.Absolute);
protected override string? BuildSourceLinkUrl(Uri contentUri, Uri gitUri, string relativeUrl, string revisionId, ITaskItem? hostItem)
{
if (!AzureDevOpsUrlParser.TryParseHostedHttp(gitUri.GetHost(), relativeUrl, out var projectPath, out var repositoryName))
{
Log.LogError(CommonResources.ValueOfWithIdentityIsInvalid, Names.SourceRoot.RepositoryUrlFullName, SourceRoot.ItemSpec, gitUri);
return null;
}
return
UriUtilities.Combine(
UriUtilities.Combine(contentUri.ToString(), projectPath), $"_apis/git/repositories/{repositoryName}/items") +
$"?api-version=1.0&versionType=commit&version={revisionId}&path=/*";
}
// TODO: confirm design and test https://github.com/dotnet/sourcelink/issues/2
private Dictionary<Uri, Uri>? TryGetEnvironmentUriMap()
{
var urlSeparators = new[] { Path.PathSeparator };
Dictionary<Uri, Uri>? map = null;
bool parse(string urls)
{
var items = urls.Split(urlSeparators, StringSplitOptions.None);
if (items.Length % 2 != 0)
{
return false;
}
for (int i = 0; i < items.Length; i += 2)
{
string originalUrl = items[i];
string mappedUrl = items[i + 1];
if (!Uri.TryCreate(originalUrl, UriKind.Absolute, out var originalUri) || originalUri.Query != "")
{
return false;
}
if (!Uri.TryCreate(mappedUrl, UriKind.Absolute, out var mappedUri) || mappedUri.Query != "")
{
return false;
}
if (map == null)
{
map = new Dictionary<Uri, Uri>();
}
map[originalUri] = mappedUri;
}
return true;
}
IEnumerable<KeyValuePair<string, string>> enumerateVariables()
{
int i = 0;
while (true)
{
var name = UrlMapEnvironmentVariableName + (i == 0 ? "" : i.ToString());
var value = Environment.GetEnvironmentVariable(name);
if (string.IsNullOrEmpty(value))
{
yield break;
}
yield return new KeyValuePair<string, string>(name, value);
i++;
}
}
foreach (var variable in enumerateVariables())
{
if (!parse(variable.Value))
{
Log.LogError(Resources.EnvironmentVariableIsNotAlistOfUrlPairs, variable.Key, variable.Value);
return null;
}
}
return map;
}
}
}