diff --git a/src/AccountDeleter/Configuration/GalleryConfiguration.cs b/src/AccountDeleter/Configuration/GalleryConfiguration.cs
index 11a99cb66e..d6a95bea8a 100644
--- a/src/AccountDeleter/Configuration/GalleryConfiguration.cs
+++ b/src/AccountDeleter/Configuration/GalleryConfiguration.cs
@@ -115,5 +115,6 @@ public string SiteRoot
public int? MaxIoThreads { get => throw new NotImplementedException(); set => throw new NotImplementedException(); }
public string InternalMicrosoftTenantKey { get => throw new NotImplementedException(); set => throw new NotImplementedException(); }
public string AdminSenderUser { get => throw new NotImplementedException(); set => throw new NotImplementedException(); }
+ public string SupportEmailSiteRoot { get => throw new NotImplementedException(); set => throw new NotImplementedException(); }
}
}
diff --git a/src/AccountDeleter/Providers/AccountDeleteUrlHelper.cs b/src/AccountDeleter/Providers/AccountDeleteUrlHelper.cs
index d44ed96636..3624f1d84c 100644
--- a/src/AccountDeleter/Providers/AccountDeleteUrlHelper.cs
+++ b/src/AccountDeleter/Providers/AccountDeleteUrlHelper.cs
@@ -7,7 +7,7 @@ namespace NuGetGallery.AccountDeleter
{
public class AccountDeleteUrlHelper : IUrlHelper
{
- public string ConfirmPendingOwnershipRequest(string id, string username, string confirmationCode, bool relativeUrl)
+ public string ConfirmPendingOwnershipRequest(string id, string username, string confirmationCode, bool relativeUrl, bool supportEmail)
{
throw new NotImplementedException();
}
@@ -17,12 +17,12 @@ public string ManagePackageOwnership(string id, bool relativeUrl)
throw new NotImplementedException();
}
- public string Package(string id, string version, bool relativeUrl)
+ public string Package(string id, string version, bool relativeUrl, bool supportEmail)
{
throw new NotImplementedException();
}
- public string RejectPendingOwnershipRequest(string id, string username, string confirmationCode, bool relativeUrl)
+ public string RejectPendingOwnershipRequest(string id, string username, string confirmationCode, bool relativeUrl, bool supportEmail)
{
throw new NotImplementedException();
}
diff --git a/src/GalleryTools/App.config b/src/GalleryTools/App.config
index 11892f2d4e..b1dc6b78f3 100644
--- a/src/GalleryTools/App.config
+++ b/src/GalleryTools/App.config
@@ -25,6 +25,7 @@
+
diff --git a/src/NuGetGallery.Services/Configuration/AppConfiguration.cs b/src/NuGetGallery.Services/Configuration/AppConfiguration.cs
index a03ee49846..2eef4c2632 100644
--- a/src/NuGetGallery.Services/Configuration/AppConfiguration.cs
+++ b/src/NuGetGallery.Services/Configuration/AppConfiguration.cs
@@ -210,6 +210,11 @@ public class AppConfiguration : IAppConfiguration
///
public string SiteRoot { get; set; }
+ ///
+ /// Gets the protocol-independent support email site root
+ ///
+ public string SupportEmailSiteRoot { get; set; }
+
///
/// Private key for verifying recaptcha user response.
///
diff --git a/src/NuGetGallery.Services/Configuration/ConfigurationService.cs b/src/NuGetGallery.Services/Configuration/ConfigurationService.cs
index d55706f978..fba91fe698 100644
--- a/src/NuGetGallery.Services/Configuration/ConfigurationService.cs
+++ b/src/NuGetGallery.Services/Configuration/ConfigurationService.cs
@@ -26,6 +26,7 @@ public class ConfigurationService : IGalleryConfigurationService, IConfiguration
private readonly Lazy _httpSiteRootThunk;
private readonly Lazy _httpsSiteRootThunk;
+ private readonly Lazy _httpsEmailSupportSiteRootThunk;
private readonly Lazy _lazyAppConfiguration;
private readonly Lazy _lazyFeatureConfiguration;
private readonly Lazy _lazyServiceBusConfiguration;
@@ -59,6 +60,7 @@ public ConfigurationService()
{
_httpSiteRootThunk = new Lazy(GetHttpSiteRoot);
_httpsSiteRootThunk = new Lazy(GetHttpsSiteRoot);
+ _httpsEmailSupportSiteRootThunk = new Lazy(GetHttpsSupportEmailSiteRoot);
_lazyAppConfiguration = new Lazy(() => ResolveSettings().Result);
_lazyFeatureConfiguration = new Lazy(() => ResolveFeatures().Result);
@@ -89,6 +91,15 @@ public string GetSiteRoot(bool useHttps)
return useHttps ? _httpsSiteRootThunk.Value : _httpSiteRootThunk.Value;
}
+ ///
+ /// Gets the support email site root using the specified protocol
+ ///
+ ///
+ public string GetSupportEmailSiteRoot()
+ {
+ return _httpsEmailSupportSiteRootThunk.Value;
+ }
+
public Task Get() where T : NuGet.Services.Configuration.Configuration, new()
{
// Get the prefix specified by the ConfigurationKeyPrefixAttribute on the class if it exists.
@@ -209,19 +220,7 @@ private string GetHttpSiteRoot()
{
var siteRoot = Current.SiteRoot;
- if (siteRoot == null)
- {
- // No SiteRoot configured in settings.
- // Fallback to detected site root.
- var request = GetCurrentRequest();
- siteRoot = request.Url.GetLeftPart(UriPartial.Authority) + '/';
- }
-
- if (!siteRoot.StartsWith("http://", StringComparison.OrdinalIgnoreCase)
- && !siteRoot.StartsWith("https://", StringComparison.OrdinalIgnoreCase))
- {
- throw new InvalidOperationException("The configured site root must start with either http:// or https://.");
- }
+ CheckValidSiteRoot(siteRoot);
if (siteRoot.StartsWith("https://", StringComparison.OrdinalIgnoreCase))
{
@@ -242,5 +241,36 @@ private string GetHttpsSiteRoot()
return "https://" + siteRoot.Substring(7);
}
+
+ private string GetHttpsSupportEmailSiteRoot()
+ {
+ var siteRoot = Current.SupportEmailSiteRoot;
+
+ CheckValidSiteRoot(siteRoot);
+
+ if (siteRoot.StartsWith("http://", StringComparison.OrdinalIgnoreCase))
+ {
+ siteRoot = "https://" + siteRoot.Substring(7);
+ }
+
+ return siteRoot;
+ }
+
+ private void CheckValidSiteRoot(string siteRoot)
+ {
+ if (siteRoot == null)
+ {
+ // No SiteRoot configured in settings.
+ // Fallback to detected site root.
+ var request = GetCurrentRequest();
+ siteRoot = request.Url.GetLeftPart(UriPartial.Authority) + '/';
+ }
+
+ if (!siteRoot.StartsWith("http://", StringComparison.OrdinalIgnoreCase)
+ && !siteRoot.StartsWith("https://", StringComparison.OrdinalIgnoreCase))
+ {
+ throw new InvalidOperationException("The configured site root must start with either http:// or https://.");
+ }
+ }
}
}
\ No newline at end of file
diff --git a/src/NuGetGallery.Services/Configuration/IAppConfiguration.cs b/src/NuGetGallery.Services/Configuration/IAppConfiguration.cs
index 0e30dbd6bb..01f0219939 100644
--- a/src/NuGetGallery.Services/Configuration/IAppConfiguration.cs
+++ b/src/NuGetGallery.Services/Configuration/IAppConfiguration.cs
@@ -231,6 +231,11 @@ public interface IAppConfiguration : IMessageServiceConfiguration
///
string SiteRoot { get; set; }
+ ///
+ /// Gets the protocol-independent support email site root
+ ///
+ string SupportEmailSiteRoot { get; set; }
+
///
/// Private key for verifying recaptcha user response.
///
diff --git a/src/NuGetGallery.Services/Configuration/IGalleryConfigurationService.cs b/src/NuGetGallery.Services/Configuration/IGalleryConfigurationService.cs
index 07260407e2..8dbc1b68bc 100644
--- a/src/NuGetGallery.Services/Configuration/IGalleryConfigurationService.cs
+++ b/src/NuGetGallery.Services/Configuration/IGalleryConfigurationService.cs
@@ -17,6 +17,11 @@ public interface IGalleryConfigurationService
/// If true, the root will be returned in HTTPS form, otherwise, HTTP.
string GetSiteRoot(bool useHttps);
+ ///
+ /// Gets the support email site root using the specified protocol
+ ///
+ string GetSupportEmailSiteRoot();
+
///
/// Populate the properties of from configuration.
///
diff --git a/src/NuGetGallery.Services/PackageManagement/PackageOwnershipManagementService.cs b/src/NuGetGallery.Services/PackageManagement/PackageOwnershipManagementService.cs
index 3ea8e4f596..9c27e6d971 100644
--- a/src/NuGetGallery.Services/PackageManagement/PackageOwnershipManagementService.cs
+++ b/src/NuGetGallery.Services/PackageManagement/PackageOwnershipManagementService.cs
@@ -49,7 +49,7 @@ public async Task AddPackageOwnerWithMessagesAsync(PackageRegistration packageRe
{
await AddPackageOwnerAsync(packageRegistration, user, commitChanges: true);
- var packageUrl = _urlHelper.Package(packageRegistration.Id, version: null, relativeUrl: false);
+ var packageUrl = _urlHelper.Package(packageRegistration.Id, version: null, relativeUrl: false, supportEmail: true);
// Accumulate the tasks so that they are sent in parallel and as many messages as possible are sent even if
// one fails (i.e. throws an exception).
@@ -155,7 +155,7 @@ public async Task AddPackageOwnershipRequestWithMessagesAsy
var encodedMessage = HttpUtility.HtmlEncode(message ?? string.Empty);
- var packageUrl = _urlHelper.Package(packageRegistration.Id, version: null, relativeUrl: false);
+ var packageUrl = _urlHelper.Package(packageRegistration.Id, version: null, relativeUrl: false, supportEmail: true);
var ownerRequest = await AddPackageOwnershipRequestAsync(
packageRegistration, requestingOwner, newOwner);
@@ -164,13 +164,15 @@ public async Task AddPackageOwnershipRequestWithMessagesAsy
packageRegistration.Id,
newOwner.Username,
ownerRequest.ConfirmationCode,
- relativeUrl: false);
+ relativeUrl: false,
+ supportEmail: true);
var rejectionUrl = _urlHelper.RejectPendingOwnershipRequest(
packageRegistration.Id,
newOwner.Username,
ownerRequest.ConfirmationCode,
- relativeUrl: false);
+ relativeUrl: false,
+ supportEmail: true);
var manageUrl = _urlHelper.ManagePackageOwnership(
packageRegistration.Id,
diff --git a/src/NuGetGallery.Services/Providers/IUrlHelper.cs b/src/NuGetGallery.Services/Providers/IUrlHelper.cs
index 1bc927fb49..3894eb63df 100644
--- a/src/NuGetGallery.Services/Providers/IUrlHelper.cs
+++ b/src/NuGetGallery.Services/Providers/IUrlHelper.cs
@@ -17,8 +17,9 @@ public interface IUrlHelper
/// The package ID to link to.
/// The specific package version to link to. Can be null.
/// True to return a relative URL, false to return an absolute URL.
+ /// True to return a supportEmail root site URL, false to return an root site URL.
/// The relative or absolute URL as a string.
- string Package(string id, string version, bool relativeUrl);
+ string Package(string id, string version, bool relativeUrl, bool supportEmail);
///
/// Produces a URL to the package ownership request confirmation page.
@@ -27,8 +28,9 @@ public interface IUrlHelper
/// The username of the ownership request recipient (new owner).
/// The confirmation code (secret) associated with the request.
/// True to return a relative URL, false to return an absolute URL.
+ /// True to return a supportEmail root site URL, false to return an root site URL.
/// The relative or absolute URL as a string.
- string ConfirmPendingOwnershipRequest(string id, string username, string confirmationCode, bool relativeUrl);
+ string ConfirmPendingOwnershipRequest(string id, string username, string confirmationCode, bool relativeUrl, bool supportEmail);
///
/// Produces a URL to the package ownership request rejection page.
@@ -37,8 +39,9 @@ public interface IUrlHelper
/// The username of the ownership request recipient (new owner).
/// The confirmation code (secret) associated with the request.
/// True to return a relative URL, false to return an absolute URL.
+ /// True to return a supportEmail root site URL, false to return an root site URL.
/// The relative or absolute URL as a string.
- string RejectPendingOwnershipRequest(string id, string username, string confirmationCode, bool relativeUrl);
+ string RejectPendingOwnershipRequest(string id, string username, string confirmationCode, bool relativeUrl, bool supportEmail);
///
/// Produces a URL to manage the ownership of an existing package.
diff --git a/src/NuGetGallery/Areas/Admin/Controllers/ApiKeysController.cs b/src/NuGetGallery/Areas/Admin/Controllers/ApiKeysController.cs
index 24bdd779dc..b90d4b3645 100644
--- a/src/NuGetGallery/Areas/Admin/Controllers/ApiKeysController.cs
+++ b/src/NuGetGallery/Areas/Admin/Controllers/ApiKeysController.cs
@@ -130,8 +130,8 @@ public async Task Revoke(RevokeApiKeysRequest revokeApiKeysRequest
credential: apiKeyCredential,
leakedUrl: apiKeyInfo.LeakedUrl,
revocationSource: apiKeyInfo.RevocationSource,
- manageApiKeyUrl: Url.ManageMyApiKeys(relativeUrl: false),
- contactUrl: Url.Contact(relativeUrl: false));
+ manageApiKeyUrl: Url.ManageMyApiKeys(relativeUrl: false, supportEmail: true),
+ contactUrl: Url.Contact(relativeUrl: false, supportEmail: true));
await _messageService.SendMessageAsync(credentialRevokedMessage);
await _authenticationService.RevokeApiKeyCredential(apiKeyCredential, revocationSourceKey, commitChanges: false);
diff --git a/src/NuGetGallery/Controllers/AuthenticationController.cs b/src/NuGetGallery/Controllers/AuthenticationController.cs
index 5b6a633392..9bc921256a 100644
--- a/src/NuGetGallery/Controllers/AuthenticationController.cs
+++ b/src/NuGetGallery/Controllers/AuthenticationController.cs
@@ -330,7 +330,8 @@ public virtual async Task Register(LogOnViewModel model, string re
Url.ConfirmEmail(
user.User.Username,
user.User.EmailConfirmationToken,
- relativeUrl: false));
+ relativeUrl: false,
+ supportEmail: true));
await _messageService.SendMessageAsync(message);
}
diff --git a/src/NuGetGallery/Controllers/OrganizationsController.cs b/src/NuGetGallery/Controllers/OrganizationsController.cs
index 817714c0db..ae2ddc694b 100644
--- a/src/NuGetGallery/Controllers/OrganizationsController.cs
+++ b/src/NuGetGallery/Controllers/OrganizationsController.cs
@@ -152,9 +152,9 @@ public async Task AddMember(string accountName, string memberName, b
request.NewMember,
currentUser,
request.IsAdmin,
- profileUrl: Url.User(account, relativeUrl: false),
- confirmationUrl: Url.AcceptOrganizationMembershipRequest(request, relativeUrl: false),
- rejectionUrl: Url.RejectOrganizationMembershipRequest(request, relativeUrl: false));
+ profileUrl: Url.User(account, relativeUrl: false, supportEmail: true),
+ confirmationUrl: Url.AcceptOrganizationMembershipRequest(request, relativeUrl: false, supportEmail: true),
+ rejectionUrl: Url.RejectOrganizationMembershipRequest(request, relativeUrl: false, supportEmail: true));
await MessageService.SendMessageAsync(organizationMembershipRequestMessage);
var organizationMembershipRequestInitiatedMessage = new OrganizationMembershipRequestInitiatedMessage(
diff --git a/src/NuGetGallery/Controllers/UsersController.cs b/src/NuGetGallery/Controllers/UsersController.cs
index b4bc42c76c..4a5e0f7369 100644
--- a/src/NuGetGallery/Controllers/UsersController.cs
+++ b/src/NuGetGallery/Controllers/UsersController.cs
@@ -203,9 +203,9 @@ public virtual async Task TransformToOrganization(TransformAccount
_config,
accountToTransform,
adminUser,
- profileUrl: Url.User(accountToTransform, relativeUrl: false),
- confirmationUrl: Url.ConfirmTransformAccount(accountToTransform, relativeUrl: false),
- rejectionUrl: Url.RejectTransformAccount(accountToTransform, relativeUrl: false));
+ profileUrl: Url.User(accountToTransform, relativeUrl: false, supportEmail: true),
+ confirmationUrl: Url.ConfirmTransformAccount(accountToTransform, relativeUrl: false, supportEmail: true),
+ rejectionUrl: Url.RejectTransformAccount(accountToTransform, relativeUrl: false, supportEmail: true));
await MessageService.SendMessageAsync(organizationTransformRequestMessage);
var organizationTransformInitiatedMessage = new OrganizationTransformInitiatedMessage(
@@ -1257,7 +1257,8 @@ private async Task SendPasswordResetEmailAsync(User user, bool for
user.Username,
user.PasswordResetToken,
forgotPassword,
- relativeUrl: false);
+ relativeUrl: false,
+ supportEmail: true);
var message = new PasswordResetInstructionsMessage(
MessageServiceConfiguration,
diff --git a/src/NuGetGallery/UrlHelperExtensions.cs b/src/NuGetGallery/UrlHelperExtensions.cs
index fd49b5d555..eb8c36227f 100644
--- a/src/NuGetGallery/UrlHelperExtensions.cs
+++ b/src/NuGetGallery/UrlHelperExtensions.cs
@@ -8,6 +8,7 @@
using System.Web.Mvc;
using System.Web.Routing;
using NuGet.Services.Entities;
+using NuGet.Services.Logging;
using NuGetGallery.Areas.Admin;
using NuGetGallery.Areas.Admin.Controllers;
using NuGetGallery.Configuration;
@@ -68,6 +69,11 @@ internal static string GetSiteRoot(bool useHttps)
return _configuration.GetSiteRoot(useHttps);
}
+ internal static string GetSupportEmailSiteRoot(bool useHttps)
+ {
+ return _configuration.GetSupportEmailSiteRoot();
+ }
+
public static string GetCanonicalLinkUrl(this UrlHelper url)
{
var current = url.RequestContext.HttpContext.Request.Url;
@@ -89,6 +95,12 @@ private static string GetConfiguredSiteHostName()
return new Uri(siteRoot).Host;
}
+ private static string GetConfiguredSupportEmailSiteHostName()
+ {
+ var siteRoot = GetSupportEmailSiteRoot(useHttps: true);
+ return new Uri(siteRoot).Host;
+ }
+
public static string Home(this UrlHelper url, bool relativeUrl = true)
{
return GetRouteLink(url, RouteName.Home, relativeUrl);
@@ -228,9 +240,9 @@ public static RouteUrlTemplate PackageRegistrationTemplate
return new RouteUrlTemplate(linkGenerator, routesGenerator);
}
- public static string Package(this UrlHelper url, string id, bool relativeUrl = true)
+ public static string Package(this UrlHelper url, string id, bool relativeUrl = true, bool supportEmail = false)
{
- return url.Package(id, version: null, relativeUrl: relativeUrl);
+ return url.Package(id, version: null, relativeUrl: relativeUrl, supportEmail: supportEmail);
}
public static string Package(
@@ -238,7 +250,8 @@ public static string Package(
string id,
string version,
bool relativeUrl = true,
- bool preview = false)
+ bool preview = false,
+ bool supportEmail = false)
{
var normalized = (version != null) ? NuGetVersionFormatter.Normalize(version) : version;
@@ -251,7 +264,8 @@ public static string Package(
{ "id", id },
{ "version", normalized },
{ "preview", preview ? "1" : null }
- });
+ },
+ supportEmail: supportEmail);
// Ensure trailing slashes for versionless package URLs, as a fix for package filenames that look like known file extensions
return version == null ? EnsureTrailingSlash(result) : result;
@@ -592,7 +606,8 @@ public static string User(
this UrlHelper url,
User user,
int page = 1,
- bool relativeUrl = true)
+ bool relativeUrl = true,
+ bool supportEmail = false)
{
var routeValues = new RouteValueDictionary
{
@@ -604,7 +619,7 @@ public static string User(
routeValues.Add("page", page);
}
- return GetActionLink(url, "Profiles", "Users", relativeUrl, routeValues);
+ return GetActionLink(url, "Profiles", "Users", relativeUrl, routeValues, supportEmail: supportEmail);
}
public static string Avatar(
@@ -965,13 +980,14 @@ private static string GetAuthenticationRoute(this UrlHelper url, string action,
interceptReturnUrl: false);
}
- public static string ManageMyApiKeys(this UrlHelper url, bool relativeUrl = true)
+ public static string ManageMyApiKeys(this UrlHelper url, bool relativeUrl = true, bool supportEmail = false)
{
return GetActionLink(
url,
nameof(UsersController.ApiKeys),
"Users",
- relativeUrl);
+ relativeUrl,
+ supportEmail: supportEmail);
}
public static string ManageMyOrganizations(this UrlHelper url, bool relativeUrl = true)
@@ -1016,35 +1032,37 @@ public static string AddOrganizationMember(this UrlHelper url, string accountNam
});
}
- public static string AcceptOrganizationMembershipRequest(this UrlHelper url, MembershipRequest request, bool relativeUrl = true)
+ public static string AcceptOrganizationMembershipRequest(this UrlHelper url, MembershipRequest request, bool relativeUrl = true, bool supportEmail = false)
{
- return url.AcceptOrganizationMembershipRequest(request.Organization.Username, request.ConfirmationToken, relativeUrl);
+ return url.AcceptOrganizationMembershipRequest(request.Organization.Username, request.ConfirmationToken, relativeUrl, supportEmail);
}
- public static string RejectOrganizationMembershipRequest(this UrlHelper url, MembershipRequest request, bool relativeUrl = true)
+ public static string RejectOrganizationMembershipRequest(this UrlHelper url, MembershipRequest request, bool relativeUrl = true, bool supportEmail = false)
{
- return url.RejectOrganizationMembershipRequest(request.Organization.Username, request.ConfirmationToken, relativeUrl);
+ return url.RejectOrganizationMembershipRequest(request.Organization.Username, request.ConfirmationToken, relativeUrl, supportEmail);
}
- public static string AcceptOrganizationMembershipRequest(this UrlHelper url, string organizationUsername, string confirmationToken, bool relativeUrl = true)
+ public static string AcceptOrganizationMembershipRequest(this UrlHelper url, string organizationUsername, string confirmationToken, bool relativeUrl = true, bool supportEmail = false)
{
return url.HandleOrganizationMembershipRequest(
nameof(OrganizationsController.ConfirmMemberRequest),
organizationUsername,
confirmationToken,
- relativeUrl);
+ relativeUrl,
+ supportEmail);
}
- public static string RejectOrganizationMembershipRequest(this UrlHelper url, string organizationUsername, string confirmationToken, bool relativeUrl = true)
+ public static string RejectOrganizationMembershipRequest(this UrlHelper url, string organizationUsername, string confirmationToken, bool relativeUrl = true, bool supportEmail = false)
{
return url.HandleOrganizationMembershipRequest(
nameof(OrganizationsController.RejectMemberRequest),
organizationUsername,
confirmationToken,
- relativeUrl);
+ relativeUrl,
+ supportEmail);
}
- private static string HandleOrganizationMembershipRequest(this UrlHelper url, string actionName, string organizationUsername, string confirmationToken, bool relativeUrl = true)
+ private static string HandleOrganizationMembershipRequest(this UrlHelper url, string actionName, string organizationUsername, string confirmationToken, bool relativeUrl = true, bool supportEmail = false)
{
return GetActionLink(url,
actionName,
@@ -1054,7 +1072,8 @@ private static string HandleOrganizationMembershipRequest(this UrlHelper url, st
{
{ "accountName", organizationUsername },
{ "confirmationToken", confirmationToken }
- });
+ },
+ supportEmail: supportEmail);
}
public static string CancelOrganizationMembershipRequest(this UrlHelper url, string accountName, bool relativeUrl = true)
@@ -1199,7 +1218,8 @@ public static string ConfirmPendingOwnershipRequest(
string packageId,
string username,
string confirmationCode,
- bool relativeUrl = true)
+ bool relativeUrl = true,
+ bool supportEmail = false)
{
return HandlePendingOwnershipRequest(
url,
@@ -1207,7 +1227,8 @@ public static string ConfirmPendingOwnershipRequest(
packageId,
username,
confirmationCode,
- relativeUrl);
+ relativeUrl,
+ supportEmail);
}
public static RouteUrlTemplate RejectPendingOwnershipRequestTemplate(
@@ -1225,7 +1246,8 @@ public static string RejectPendingOwnershipRequest(
string packageId,
string username,
string confirmationCode,
- bool relativeUrl = true)
+ bool relativeUrl = true,
+ bool supportEmail = false)
{
return HandlePendingOwnershipRequest(
url,
@@ -1233,7 +1255,8 @@ public static string RejectPendingOwnershipRequest(
packageId,
username,
confirmationCode,
- relativeUrl);
+ relativeUrl,
+ supportEmail);
}
private static RouteUrlTemplate HandlePendingOwnershipRequestTemplate(
@@ -1264,7 +1287,8 @@ private static string HandlePendingOwnershipRequest(
string packageId,
string username,
string confirmationCode,
- bool relativeUrl = true)
+ bool relativeUrl = true,
+ bool supportEmail = false)
{
var routeValues = new RouteValueDictionary
{
@@ -1273,14 +1297,15 @@ private static string HandlePendingOwnershipRequest(
["token"] = confirmationCode
};
- return GetActionLink(url, actionName, "Packages", relativeUrl, routeValues);
+ return GetActionLink(url, actionName, "Packages", relativeUrl, routeValues, supportEmail: supportEmail);
}
public static string ConfirmEmail(
this UrlHelper url,
string username,
string token,
- bool relativeUrl = true)
+ bool relativeUrl = true,
+ bool supportEmail = false)
{
var routeValues = new RouteValueDictionary
{
@@ -1288,14 +1313,15 @@ public static string ConfirmEmail(
["token"] = token
};
- return GetActionLink(url, "Confirm", "Users", relativeUrl, routeValues);
+ return GetActionLink(url, "Confirm", "Users", relativeUrl, routeValues, supportEmail: supportEmail);
}
public static string ConfirmOrganizationEmail(
this UrlHelper url,
string username,
string token,
- bool relativeUrl = true)
+ bool relativeUrl = true,
+ bool supportEmail = false)
{
var routeValues = new RouteValueDictionary
{
@@ -1303,7 +1329,7 @@ public static string ConfirmOrganizationEmail(
["token"] = token
};
- return GetActionLink(url, "Confirm", "Organizations", relativeUrl, routeValues);
+ return GetActionLink(url, "Confirm", "Organizations", relativeUrl, routeValues, supportEmail: supportEmail);
}
public static string ResetEmailOrPassword(
@@ -1311,7 +1337,8 @@ public static string ResetEmailOrPassword(
string username,
string token,
bool forgotPassword,
- bool relativeUrl = true)
+ bool relativeUrl = true,
+ bool supportEmail = false)
{
var routeValues = new RouteValueDictionary
{
@@ -1320,7 +1347,7 @@ public static string ResetEmailOrPassword(
["forgot"] = forgotPassword
};
- return GetActionLink(url, "ResetPassword", "Users", relativeUrl, routeValues);
+ return GetActionLink(url, "ResetPassword", "Users", relativeUrl, routeValues, supportEmail: supportEmail);
}
public static string VerifyPackage(this UrlHelper url, bool relativeUrl = true)
@@ -1338,9 +1365,9 @@ public static string Downloads(this UrlHelper url, bool relativeUrl = true)
return GetRouteLink(url, RouteName.Downloads, relativeUrl);
}
- public static string Contact(this UrlHelper url, bool relativeUrl = true)
+ public static string Contact(this UrlHelper url, bool relativeUrl = true, bool supportEmail = false)
{
- return GetActionLink(url, "Contact", "Pages", relativeUrl);
+ return GetActionLink(url, "Contact", "Pages", relativeUrl, supportEmail: supportEmail);
}
public static string ContactOwners(this UrlHelper url, IPackageVersionModel package, bool relativeUrl = true)
@@ -1442,50 +1469,55 @@ public static string TransformAccount(this UrlHelper url, bool relativeUrl = tru
relativeUrl);
}
- public static string ConfirmTransformAccount(this UrlHelper url, User accountToTransform, bool relativeUrl = true)
+ public static string ConfirmTransformAccount(this UrlHelper url, User accountToTransform, bool relativeUrl = true, bool supportEmail = false)
{
return url.HandleTransformAccount(
nameof(UsersController.ConfirmTransformToOrganization),
accountToTransform,
- relativeUrl);
+ relativeUrl,
+ supportEmail);
}
- public static string RejectTransformAccount(this UrlHelper url, User accountToTransform, bool relativeUrl = true)
+ public static string RejectTransformAccount(this UrlHelper url, User accountToTransform, bool relativeUrl = true, bool supportEmail = false)
{
return url.HandleTransformAccount(
nameof(UsersController.RejectTransformToOrganization),
accountToTransform,
- relativeUrl);
+ relativeUrl,
+ supportEmail);
}
- private static string HandleTransformAccount(this UrlHelper url, string action, User accountToTransform, bool relativeUrl = true)
+ private static string HandleTransformAccount(this UrlHelper url, string action, User accountToTransform, bool relativeUrl = true, bool supportEmail = false)
{
return url.HandleTransformAccount(
action,
accountToTransform.Username,
accountToTransform.OrganizationMigrationRequest.ConfirmationToken,
- relativeUrl);
+ relativeUrl,
+ supportEmail);
}
- public static string ConfirmTransformAccount(this UrlHelper url, string accountToTransformUsername, string confirmationToken, bool relativeUrl = true)
+ public static string ConfirmTransformAccount(this UrlHelper url, string accountToTransformUsername, string confirmationToken, bool relativeUrl = true, bool supportEmail = false)
{
return url.HandleTransformAccount(
nameof(UsersController.ConfirmTransformToOrganization),
accountToTransformUsername,
confirmationToken,
- relativeUrl);
+ relativeUrl,
+ supportEmail);
}
- public static string RejectTransformAccount(this UrlHelper url, string accountToTransformUsername, string confirmationToken, bool relativeUrl = true)
+ public static string RejectTransformAccount(this UrlHelper url, string accountToTransformUsername, string confirmationToken, bool relativeUrl = true, bool supportEmail = false)
{
return url.HandleTransformAccount(
nameof(UsersController.RejectTransformToOrganization),
accountToTransformUsername,
confirmationToken,
- relativeUrl);
+ relativeUrl,
+ supportEmail);
}
- private static string HandleTransformAccount(this UrlHelper url, string action, string accountToTransformUsername, string confirmationToken, bool relativeUrl = true)
+ private static string HandleTransformAccount(this UrlHelper url, string action, string accountToTransformUsername, string confirmationToken, bool relativeUrl = true, bool supportEmail = false)
{
return GetActionLink(
url,
@@ -1496,7 +1528,8 @@ private static string HandleTransformAccount(this UrlHelper url, string action,
{
{ "accountNameToTransform", accountToTransformUsername },
{ "token", confirmationToken }
- });
+ },
+ supportEmail: supportEmail);
}
public static string CancelTransformAccount(this UrlHelper url, User accountToTransform, bool relativeUrl = true)
@@ -1549,11 +1582,12 @@ public static string GetActionLink(
bool relativeUrl,
RouteValueDictionary routeValues = null,
bool interceptReturnUrl = true,
- string area = "" // Default to no area. Admin links should specify the "Admin" area explicitly.
+ string area = "", // Default to no area. Admin links should specify the "Admin" area explicitly.
+ bool supportEmail = false
)
{
var protocol = GetProtocol(url);
- var hostName = GetConfiguredSiteHostName();
+ var hostName = supportEmail ? GetConfiguredSupportEmailSiteHostName(): GetConfiguredSiteHostName();
routeValues = routeValues ?? new RouteValueDictionary();
if (!routeValues.ContainsKey(Area))
@@ -1600,10 +1634,12 @@ private static string GetRouteLink(
UrlHelper url,
string routeName,
bool relativeUrl,
- RouteValueDictionary routeValues = null)
+ RouteValueDictionary routeValues = null,
+ bool supportEmail = false)
{
var protocol = GetProtocol(url);
- var hostName = GetConfiguredSiteHostName();
+
+ var hostName = supportEmail ? GetConfiguredSupportEmailSiteHostName() : GetConfiguredSiteHostName();
var routeLink = url.RouteUrl(routeName, routeValues, protocol, hostName);
diff --git a/src/NuGetGallery/UrlHelperWrapper.cs b/src/NuGetGallery/UrlHelperWrapper.cs
index 93048da74f..dd8c18c892 100644
--- a/src/NuGetGallery/UrlHelperWrapper.cs
+++ b/src/NuGetGallery/UrlHelperWrapper.cs
@@ -15,9 +15,9 @@ public UrlHelperWrapper(UrlHelper urlHelper)
_urlHelper = urlHelper ?? throw new ArgumentNullException(nameof(urlHelper));
}
- public string ConfirmPendingOwnershipRequest(string id, string username, string confirmationCode, bool relativeUrl)
+ public string ConfirmPendingOwnershipRequest(string id, string username, string confirmationCode, bool relativeUrl, bool supportEmail)
{
- return _urlHelper.ConfirmPendingOwnershipRequest(id, username, confirmationCode, relativeUrl);
+ return _urlHelper.ConfirmPendingOwnershipRequest(id, username, confirmationCode, relativeUrl, supportEmail);
}
public string ManagePackageOwnership(string id, bool relativeUrl)
@@ -25,14 +25,14 @@ public string ManagePackageOwnership(string id, bool relativeUrl)
return _urlHelper.ManagePackageOwnership(id, relativeUrl);
}
- public string Package(string id, string version, bool relativeUrl)
+ public string Package(string id, string version, bool relativeUrl, bool supportEmail)
{
- return _urlHelper.Package(id, version, relativeUrl);
+ return _urlHelper.Package(id, version, relativeUrl, supportEmail: supportEmail);
}
- public string RejectPendingOwnershipRequest(string id, string username, string confirmationCode, bool relativeUrl)
+ public string RejectPendingOwnershipRequest(string id, string username, string confirmationCode, bool relativeUrl, bool supportEmail)
{
- return _urlHelper.RejectPendingOwnershipRequest(id, username, confirmationCode, relativeUrl);
+ return _urlHelper.RejectPendingOwnershipRequest(id, username, confirmationCode, relativeUrl, supportEmail);
}
}
}
diff --git a/src/NuGetGallery/Web.config b/src/NuGetGallery/Web.config
index 56c5379009..02743e762a 100644
--- a/src/NuGetGallery/Web.config
+++ b/src/NuGetGallery/Web.config
@@ -85,6 +85,7 @@
+
@@ -673,4 +674,4 @@
-
+
\ No newline at end of file
diff --git a/tests/NuGetGallery.Facts/App_Start/ConfigurationServiceFacts.cs b/tests/NuGetGallery.Facts/App_Start/ConfigurationServiceFacts.cs
index 57b9e61b9f..88d0e8f157 100644
--- a/tests/NuGetGallery.Facts/App_Start/ConfigurationServiceFacts.cs
+++ b/tests/NuGetGallery.Facts/App_Start/ConfigurationServiceFacts.cs
@@ -22,6 +22,7 @@ private class TestableConfigurationService : ConfigurationService
public TestableConfigurationService() : base()
{
StubConfiguredSiteRoot = "http://aSiteRoot/";
+ StubConfiguredSupportEmailSiteRoot = "http://aSupportEmailSiteRoot";
StubRequest = new Mock();
StubRequest.Setup(stub => stub.IsLocal).Returns(false);
@@ -31,6 +32,7 @@ public TestableConfigurationService() : base()
}
public string StubConfiguredSiteRoot { get; set; }
+ public string StubConfiguredSupportEmailSiteRoot { get; set; }
public Mock StubRequest { get; set; }
protected override string GetAppSetting(string settingName)
@@ -42,8 +44,14 @@ protected override string GetAppSetting(string settingName)
return StubConfiguredSiteRoot;
}
+ if (settingName == $"{SettingPrefix}{nameof(tempAppConfig.SupportEmailSiteRoot)}")
+ {
+ return StubConfiguredSupportEmailSiteRoot;
+ }
+
return string.Empty;
}
+
}
[Fact]
@@ -100,6 +108,37 @@ public void WillThrowIfConfiguredSiteRootIsNotHttpOrHttps()
Assert.Throws(() => configuration.GetSiteRoot(useHttps: false));
}
+
+ [Fact]
+ public void WillGetTheConfiguredHttpsSupportEmailSiteRoot()
+ {
+ var configuration = new TestableConfigurationService();
+ configuration.StubConfiguredSupportEmailSiteRoot = "https://aSupportEmailSiteRoot";
+
+ var siteRoot = configuration.GetSupportEmailSiteRoot();
+
+ Assert.Equal("https://aSupportEmailSiteRoot", siteRoot);
+ }
+
+ [Fact]
+ public void WillThrowIfConfiguredSupportEmailSiteRootIsNotHttpOrHttps()
+ {
+ var configuration = new TestableConfigurationService();
+ configuration.StubConfiguredSupportEmailSiteRoot = "ftp://theSupportEmailSiteRoot/";
+
+ Assert.Throws(() => configuration.GetSupportEmailSiteRoot());
+ }
+
+ [Fact]
+ public void WillUseHttpsWhenConfiguredSiteRootIsHttp()
+ {
+ var configuration = new TestableConfigurationService();
+ configuration.StubConfiguredSupportEmailSiteRoot = "http://aSupportEmailSiteRoot";
+
+ var siteRoot = configuration.GetSupportEmailSiteRoot();
+
+ Assert.Equal("https://aSupportEmailSiteRoot", siteRoot);
+ }
}
public class TheReadSettingMethod
diff --git a/tests/NuGetGallery.Facts/Controllers/AuthenticationControllerFacts.cs b/tests/NuGetGallery.Facts/Controllers/AuthenticationControllerFacts.cs
index 9e8da1fefa..bfd58b850e 100644
--- a/tests/NuGetGallery.Facts/Controllers/AuthenticationControllerFacts.cs
+++ b/tests/NuGetGallery.Facts/Controllers/AuthenticationControllerFacts.cs
@@ -804,7 +804,7 @@ public async Task WillCreateAndLogInTheUserWhenNotLinking()
It.Is(
msg =>
msg.User == authUser.User
- && msg.ConfirmationUrl == TestUtility.GallerySiteRootHttps + "account/confirm/" + authUser.User.Username + "/" + authUser.User.EmailConfirmationToken),
+ && msg.ConfirmationUrl == TestUtility.GallerySupportEmailSiteRootHttps + "account/confirm/" + authUser.User.Username + "/" + authUser.User.EmailConfirmationToken),
false,
false));
@@ -892,7 +892,7 @@ public async Task WillNotAutoConfirmAndWillSendConfirmationEmailWhenNotExternalC
UserInfo = new IdentityInformation("", "", authUser.User.UnconfirmedEmailAddress, "")
});
- var confirmationUrl = TestUtility.GallerySiteRootHttps + "account/confirm/" + authUser.User.Username + "/" + authUser.User.EmailConfirmationToken;
+ var confirmationUrl = TestUtility.GallerySupportEmailSiteRootHttps + "account/confirm/" + authUser.User.Username + "/" + authUser.User.EmailConfirmationToken;
var configurationService = GetConfigurationService();
var messageService = GetMock();
messageService
@@ -960,7 +960,7 @@ public async Task WillNotAutoConfirmAndWillSendConfirmationEmailWhenModelRegiste
UserInfo = new IdentityInformation("", "", "unconfirmed@example.com", "")
});
- var confirmationUrl = TestUtility.GallerySiteRootHttps + "account/confirm/" + authUser.User.Username + "/" + authUser.User.EmailConfirmationToken;
+ var confirmationUrl = TestUtility.GallerySupportEmailSiteRootHttps + "account/confirm/" + authUser.User.Username + "/" + authUser.User.EmailConfirmationToken;
var configurationService = GetConfigurationService();
var messageService = GetMock();
messageService
@@ -1147,7 +1147,7 @@ public async Task GivenValidExternalAuth_ItCreatesAccountAndLinksCredential()
It.Is(
msg =>
msg.User == authUser.User
- && msg.ConfirmationUrl == TestUtility.GallerySiteRootHttps + "account/confirm/" + authUser.User.Username + "/" + authUser.User.EmailConfirmationToken),
+ && msg.ConfirmationUrl == TestUtility.GallerySupportEmailSiteRootHttps + "account/confirm/" + authUser.User.Username + "/" + authUser.User.EmailConfirmationToken),
false,
false));
diff --git a/tests/NuGetGallery.Facts/Controllers/UsersControllerFacts.cs b/tests/NuGetGallery.Facts/Controllers/UsersControllerFacts.cs
index e884659436..88b3ac0863 100644
--- a/tests/NuGetGallery.Facts/Controllers/UsersControllerFacts.cs
+++ b/tests/NuGetGallery.Facts/Controllers/UsersControllerFacts.cs
@@ -1609,7 +1609,7 @@ public async Task GivenNoOldPassword_ItSendsAPasswordSetEmail()
await controller.ChangePassword(new UserAccountViewModel());
// Assert
- Assert.Equal(TestUtility.GallerySiteRootHttps + "account/setpassword/test/t0k3n", actualConfirmUrl);
+ Assert.Equal(TestUtility.GallerySupportEmailSiteRootHttps + "account/setpassword/test/t0k3n", actualConfirmUrl);
GetMock().VerifyAll();
}
diff --git a/tests/NuGetGallery.Facts/Framework/UnitTestBindings.cs b/tests/NuGetGallery.Facts/Framework/UnitTestBindings.cs
index 873421f80c..69cbdea97d 100644
--- a/tests/NuGetGallery.Facts/Framework/UnitTestBindings.cs
+++ b/tests/NuGetGallery.Facts/Framework/UnitTestBindings.cs
@@ -131,6 +131,7 @@ private static IGalleryConfigurationService CreateTestConfigurationService()
// We configure HTTP site root, but require SSL.
var configurationService = new TestGalleryConfigurationService();
configurationService.Current.SiteRoot = TestUtility.GallerySiteRootHttp;
+ configurationService.Current.SupportEmailSiteRoot = TestUtility.GallerySupportEmailSiteRootHttps;
configurationService.Current.RequireSSL = true;
configurationService.Current.GalleryOwner = new MailAddress("support@example.com");
diff --git a/tests/NuGetGallery.Facts/TestUtils/TestUtility.cs b/tests/NuGetGallery.Facts/TestUtils/TestUtility.cs
index c55609672b..ac027e798a 100644
--- a/tests/NuGetGallery.Facts/TestUtils/TestUtility.cs
+++ b/tests/NuGetGallery.Facts/TestUtils/TestUtility.cs
@@ -21,6 +21,7 @@ public static class TestUtility
public static readonly string GallerySiteRootHttp = $"http://{galleryHostName}/";
public static readonly string GallerySiteRootHttps = $"https://{galleryHostName}/";
+ public static readonly string GallerySupportEmailSiteRootHttps = $"https://{galleryHostName}/";
public static readonly string FakeUserName = "theUsername";
public static readonly int FakeUserKey = _key++;