User registration now actually works

Also package updates
This commit is contained in:
Thomas Hounsell 2017-04-12 15:04:23 +01:00
parent 74cbe3cb22
commit 80642a2402
39 changed files with 1600 additions and 954 deletions

View File

@ -0,0 +1,133 @@
using System;
namespace MongoAuth
{
/* Based off http://stackoverflow.com/a/7135008 */
internal static class Base32Encoding
{
public static byte[] ToBytes(string input)
{
if (string.IsNullOrEmpty(input))
{
throw new ArgumentNullException(nameof(input));
}
input = input.TrimEnd('='); //remove padding characters
int byteCount = input.Length * 5 / 8; //this must be TRUNCATED
var returnArray = new byte[byteCount];
byte curByte = 0,
bitsRemaining = 8;
int arrayIndex = 0;
foreach (char c in input)
{
int cValue = CharToValue(c);
int mask;
if (bitsRemaining > 5)
{
mask = cValue << (bitsRemaining - 5);
curByte = (byte)(curByte | mask);
bitsRemaining -= 5;
}
else
{
mask = cValue >> (5 - bitsRemaining);
curByte = (byte)(curByte | mask);
returnArray[arrayIndex++] = curByte;
curByte = (byte)(cValue << (3 + bitsRemaining));
bitsRemaining += 3;
}
}
//if we didn't end with a full byte
if (arrayIndex != byteCount)
{
returnArray[arrayIndex] = curByte;
}
return returnArray;
}
public static string ToString(byte[] input)
{
if (input == null || input.Length == 0)
{
throw new ArgumentNullException(nameof(input));
}
int charCount = (int)Math.Ceiling(input.Length / 5d) * 8;
var returnArray = new char[charCount];
byte nextChar = 0,
bitsRemaining = 5;
int arrayIndex = 0;
foreach (byte b in input)
{
nextChar = (byte)(nextChar | (b >> (8 - bitsRemaining)));
returnArray[arrayIndex++] = ValueToChar(nextChar);
if (bitsRemaining < 4)
{
nextChar = (byte)((b >> (3 - bitsRemaining)) & 31);
returnArray[arrayIndex++] = ValueToChar(nextChar);
bitsRemaining += 5;
}
bitsRemaining -= 3;
nextChar = (byte)((b << bitsRemaining) & 31);
}
//if we didn't end with a full char
if (arrayIndex != charCount)
{
returnArray[arrayIndex++] = ValueToChar(nextChar);
while (arrayIndex != charCount)
returnArray[arrayIndex++] = '='; //padding
}
return new string(returnArray);
}
private static int CharToValue(char c)
{
int value = c;
//65-90 == uppercase letters
if (value < 91 && value > 64)
{
return value - 65;
}
//50-55 == numbers 2-7
if (value < 56 && value > 49)
{
return value - 24;
}
//97-122 == lowercase letters
if (value < 123 && value > 96)
{
return value - 97;
}
throw new ArgumentException("Character is not a Base32 character.", nameof(c));
}
private static char ValueToChar(byte b)
{
if (b < 26)
{
return (char)(b + 65);
}
if (b < 32)
{
return (char)(b + 24);
}
throw new ArgumentException("Byte is not a value Base32 value.", nameof(b));
}
}
}

View File

@ -16,13 +16,13 @@ static DatabaseConfig()
? ConfigurationManager.AppSettings["data:MongoHost"]
: "localhost";
int _port;
bool success = int.TryParse(ConfigurationManager.AppSettings["data:MongoPort"], out _port);
int port;
bool success = int.TryParse(ConfigurationManager.AppSettings["data:MongoPort"], out port);
if (!success)
{
_port = 27017; // mongo default port
port = 27017; // mongo default port
}
Port = _port;
Port = port;
Database = !string.IsNullOrEmpty(ConfigurationManager.AppSettings["data:MongoDB"])
? ConfigurationManager.AppSettings["data:MongoDB"]

View File

@ -56,6 +56,7 @@
<Reference Include="System.Xml" />
</ItemGroup>
<ItemGroup>
<Compile Include="Base32Encoding.cs" />
<Compile Include="DatabaseConfig.cs" />
<Compile Include="MongoMembershipProvider.cs" />
<Compile Include="Properties\AssemblyInfo.cs" />

View File

@ -1,6 +1,7 @@
using System;
using System.Collections.Generic;
using System.Collections.Specialized;
using System.Configuration;
using System.Linq;
using System.Security.Cryptography;
using System.Text;
@ -152,9 +153,9 @@ public override MembershipUser CreateUser(string username, string password, stri
PassHash = hash,
PassSalt = salt,
EmailAddress = email,
IsApproved = false,
IsApproved = isApproved,
IsLockedOut = false,
CreationDate = DateTime.Now,
CreationDate = DateTime.UtcNow,
LastLoginDate = DateTime.MinValue,
LastActivityDate = DateTime.MinValue,
LastLockoutDate = DateTime.MinValue
@ -182,7 +183,7 @@ public override bool DeleteUser(string username, bool deleteAllRelatedData)
Task<DeleteResult> task = _memberCollection.DeleteOneAsync(m => m.UserName.ToLower() == username.ToLower());
task.Wait();
return task.Result.IsAcknowledged && (task.Result.DeletedCount == 1);
return task.Result.IsAcknowledged && task.Result.DeletedCount == 1;
}
public override MembershipUserCollection FindUsersByEmail(string emailToMatch, int pageIndex, int pageSize, out int totalRecords)
@ -211,7 +212,7 @@ public override MembershipUserCollection GetAllUsers(int pageIndex, int pageSize
foreach (MongoMember mm in pageItemsTask.Result)
{
muc.Add(new MembershipUser(Name, mm.UserName, mm.Id, mm.EmailAddress, "", "", mm.IsApproved, mm.IsLockedOut, mm.CreationDate, mm.LastLoginDate, mm.LastActivityDate, DateTime.MinValue, mm.LastLockoutDate));
muc.Add(new MembershipUser(Name, mm.UserName, mm.Id, mm.EmailAddress, "", "", mm.IsApproved, mm.IsLockedOut, FixupDatesFromMongo(mm.CreationDate), FixupDatesFromMongo(mm.LastLoginDate), FixupDatesFromMongo(mm.LastActivityDate), DateTime.MinValue, FixupDatesFromMongo(mm.LastLockoutDate)));
}
return muc;
@ -236,7 +237,7 @@ public override MembershipUser GetUser(string username, bool userIsOnline)
return mm == null
? null
: new MembershipUser(Name, mm.UserName, mm.Id, mm.EmailAddress, "", "", mm.IsApproved, mm.IsLockedOut, mm.CreationDate, mm.LastLoginDate, mm.LastActivityDate, DateTime.MinValue, mm.LastLockoutDate);
: new MembershipUser(Name, mm.UserName, mm.Id, mm.EmailAddress, "", "", mm.IsApproved, mm.IsLockedOut, FixupDatesFromMongo(mm.CreationDate), FixupDatesFromMongo(mm.LastLoginDate), FixupDatesFromMongo(mm.LastActivityDate), DateTime.MinValue, FixupDatesFromMongo(mm.LastLockoutDate));
}
public override MembershipUser GetUser(object providerUserKey, bool userIsOnline)
@ -248,7 +249,7 @@ public override MembershipUser GetUser(object providerUserKey, bool userIsOnline
return mm == null
? null
: new MembershipUser(Name, mm.UserName, mm.Id, mm.EmailAddress, "", "", mm.IsApproved, mm.IsLockedOut, mm.CreationDate, mm.LastLoginDate, mm.LastActivityDate, DateTime.MinValue, mm.LastLockoutDate);
: new MembershipUser(Name, mm.UserName, mm.Id, mm.EmailAddress, "", "", mm.IsApproved, mm.IsLockedOut, FixupDatesFromMongo(mm.CreationDate), FixupDatesFromMongo(mm.LastLoginDate), FixupDatesFromMongo(mm.LastActivityDate), DateTime.MinValue, FixupDatesFromMongo(mm.LastLockoutDate));
}
public override string GetUserNameByEmail(string email)
@ -279,7 +280,7 @@ public void ChangeLockStatus(Guid id, bool newStatus)
if (newStatus)
{
updateDefinition.Add(Builders<MongoMember>.Update.Set(u => u.LastLockoutDate, DateTime.Now));
updateDefinition.Add(Builders<MongoMember>.Update.Set(u => u.LastLockoutDate, DateTime.UtcNow));
}
else
{
@ -296,7 +297,7 @@ public override bool UnlockUser(string userName)
Task<UpdateResult> task = _memberCollection.UpdateOneAsync(Builders<MongoMember>.Filter.Eq(m => m.UserName.ToLower(), userName.ToLower()), Builders<MongoMember>.Update.Set(m => m.IsLockedOut, false));
task.Wait();
return task.Result.IsAcknowledged && (task.Result.ModifiedCount == 1);
return task.Result.IsAcknowledged && task.Result.ModifiedCount == 1;
}
public override void UpdateUser(MembershipUser user)
@ -310,7 +311,7 @@ public override bool ValidateUser(string username, string password)
task.Wait();
MongoMember mm = task.Result;
if ((mm == null)
if (mm == null
|| !(mm.IsApproved && !mm.IsLockedOut))
{
return false;
@ -331,12 +332,12 @@ public override bool ValidateUser(string username, string password)
{
if (mm.LockoutWindowStart == DateTime.MinValue)
{
mm.LockoutWindowStart = DateTime.Now;
mm.LockoutWindowStart = DateTime.UtcNow;
mm.LockoutWindowAttempts = 1;
}
else
{
if (mm.LockoutWindowStart.AddMinutes(PasswordAttemptWindow) > DateTime.Now)
if (mm.LockoutWindowStart.AddMinutes(PasswordAttemptWindow) > DateTime.UtcNow)
{
// still within window
mm.LockoutWindowAttempts++;
@ -348,14 +349,14 @@ public override bool ValidateUser(string username, string password)
else
{
// outside of window, reset
mm.LockoutWindowStart = DateTime.Now;
mm.LockoutWindowStart = DateTime.UtcNow;
mm.LockoutWindowAttempts = 1;
}
}
}
else
{
mm.LastLoginDate = DateTime.Now;
mm.LastLoginDate = DateTime.UtcNow;
mm.LockoutWindowStart = DateTime.MinValue;
mm.LockoutWindowAttempts = 0;
}
@ -366,6 +367,49 @@ public override bool ValidateUser(string username, string password)
return !isFail;
}
public async Task<string> GenerateValidationHash(Guid id)
{
MongoMember mm = await _memberCollection.Find(Builders<MongoMember>.Filter.Eq(u => u.Id, id)).FirstOrDefaultAsync();
if (mm == null)
{
return null;
}
using (SHA256 sha = SHA256.Create())
{
string content = $"{mm.Id}.{mm.PassSalt}.{ConfigurationManager.AppSettings["data:SecretKey"]}";
byte[] hashBytes = sha.ComputeHash(Encoding.UTF8.GetBytes(content));
return Base32Encoding.ToString(hashBytes);
}
}
public async Task<bool> ValidateUserFromHash(Guid id, string validate)
{
MongoMember mm = await _memberCollection.Find(Builders<MongoMember>.Filter.Eq(u => u.Id, id)).FirstOrDefaultAsync();
if (mm == null)
{
return false;
}
using (SHA256 sha = SHA256.Create())
{
string content = $"{mm.Id}.{mm.PassSalt}.{ConfigurationManager.AppSettings["data:SecretKey"]}";
byte[] hashBytes = sha.ComputeHash(Encoding.UTF8.GetBytes(content));
string expected = Base32Encoding.ToString(hashBytes);
bool success = string.Equals(expected, validate, StringComparison.InvariantCultureIgnoreCase);
if (success)
{
ChangeApproval(id, true);
}
return success;
}
}
private static byte[] CalculateHash(string password, ref byte[] salt)
{
if (!salt.Any(v => v != 0))
@ -404,6 +448,12 @@ private static int TryReadInt(string config, int defaultValue)
? temp
: defaultValue;
}
private static DateTime FixupDatesFromMongo(DateTime dt)
{
DateTime local = DateTime.SpecifyKind(dt, DateTimeKind.Local);
return local;
}
}
public class MongoMember

View File

@ -83,13 +83,20 @@ where newUsers.All(v => v != u.Id)
public override void CreateRole(string roleName)
{
MongoRole r = new MongoRole
Task<MongoRole> role = _roleCollection.Find(r => r.RoleName == roleName).SingleOrDefaultAsync();
role.Wait();
if (role.Result != null)
{
return;
}
MongoRole mr = new MongoRole
{
Id = Guid.NewGuid(),
RoleName = roleName
};
Task task = _roleCollection.InsertOneAsync(r);
Task task = _roleCollection.InsertOneAsync(mr);
task.Wait();
}
@ -98,8 +105,8 @@ public override bool DeleteRole(string roleName, bool throwOnPopulatedRole)
Task<MongoRole> role = _roleCollection.Find(r => r.RoleName == roleName).SingleOrDefaultAsync();
role.Wait();
if ((role.Result != null)
&& (role.Result.Users.Length > 0)
if (role.Result != null
&& role.Result.Users.Length > 0
&& throwOnPopulatedRole)
{
throw new ProviderException(Resources.RoleNotEmpty);
@ -145,7 +152,7 @@ public override string[] GetRolesForUser(string username)
return Array.Empty<string>();
}
Task<List<MongoRole>> role = _roleCollection.Find(r => (r.Users != null) && r.Users.Contains(user.Result.Id)).ToListAsync();
Task<List<MongoRole>> role = _roleCollection.Find(r => r.Users != null && r.Users.Contains(user.Result.Id)).ToListAsync();
role.Wait();
return role.Result.Select(r => r.RoleName).ToArray();
@ -174,8 +181,8 @@ public override bool IsUserInRole(string username, string roleName)
user.Wait();
role.Wait();
if ((user.Result == null)
|| (role.Result?.Users == null))
if (user.Result == null
|| role.Result?.Users == null)
{
return false;
}

View File

@ -186,6 +186,15 @@ public static string Common_Admin {
}
}
/// <summary>
/// Looks up a localized string similar to Change password.
/// </summary>
public static string Common_ChangePassword {
get {
return ResourceManager.GetString("Common_ChangePassword", resourceCulture);
}
}
/// <summary>
/// Looks up a localized string similar to Contribute on {0}.
/// </summary>
@ -330,6 +339,32 @@ public static string Common_Twitter {
}
}
/// <summary>
/// Looks up a localized string similar to Thank you for registering to {0}.
///
///Please verify your email address by clicking the link below, or by copying and pasting it into your browser.
///{1}
///
///If you did not register to {0}, you can ignore this email.
///
///Thanks,
///The {0} Team..
/// </summary>
public static string Email_Registration_Body {
get {
return ResourceManager.GetString("Email_Registration_Body", resourceCulture);
}
}
/// <summary>
/// Looks up a localized string similar to {0}: Please verify your email address.
/// </summary>
public static string Email_Registration_Subject {
get {
return ResourceManager.GetString("Email_Registration_Subject", resourceCulture);
}
}
/// <summary>
/// Looks up a localized string similar to About.
/// </summary>
@ -924,15 +959,6 @@ public static string Search_Year {
}
}
/// <summary>
/// Looks up a localized string similar to Every account is validated by an administrator, so be patient and check again later.
/// </summary>
public static string Support_AccountValidation {
get {
return ResourceManager.GetString("Support_AccountValidation", resourceCulture);
}
}
/// <summary>
/// Looks up a localized string similar to Additions to BuildFeed.
/// </summary>
@ -987,6 +1013,24 @@ public static string Support_EmailAddress {
}
}
/// <summary>
/// Looks up a localized string similar to An validation link has been sent to your email address. Please click the link in the email to verify the account..
/// </summary>
public static string Support_EmailValidationContent {
get {
return ResourceManager.GetString("Support_EmailValidationContent", resourceCulture);
}
}
/// <summary>
/// Looks up a localized string similar to Please verify your email address.
/// </summary>
public static string Support_EmailValidationTitle {
get {
return ResourceManager.GetString("Support_EmailValidationTitle", resourceCulture);
}
}
/// <summary>
/// Looks up a localized string similar to Enter current password.
/// </summary>
@ -1014,6 +1058,51 @@ public static string Support_EnterPassword {
}
}
/// <summary>
/// Looks up a localized string similar to An error occurred when attempting to change your password..
/// </summary>
public static string Support_Error_ChangingPasswordFail {
get {
return ResourceManager.GetString("Support_Error_ChangingPasswordFail", resourceCulture);
}
}
/// <summary>
/// Looks up a localized string similar to A user account with this email address already exists..
/// </summary>
public static string Support_Error_DuplicateEmail {
get {
return ResourceManager.GetString("Support_Error_DuplicateEmail", resourceCulture);
}
}
/// <summary>
/// Looks up a localized string similar to A user account with this user name already exists..
/// </summary>
public static string Support_Error_DuplicateUserName {
get {
return ResourceManager.GetString("Support_Error_DuplicateUserName", resourceCulture);
}
}
/// <summary>
/// Looks up a localized string similar to The password you have entered is invalid..
/// </summary>
public static string Support_Error_InvalidPassword {
get {
return ResourceManager.GetString("Support_Error_InvalidPassword", resourceCulture);
}
}
/// <summary>
/// Looks up a localized string similar to An unknown error has occurred..
/// </summary>
public static string Support_Error_UnknownError {
get {
return ResourceManager.GetString("Support_Error_UnknownError", resourceCulture);
}
}
/// <summary>
/// Looks up a localized string similar to Highest version.
/// </summary>
@ -1095,15 +1184,6 @@ public static string Support_RememberMe {
}
}
/// <summary>
/// Looks up a localized string similar to Thank you for registering.
/// </summary>
public static string Support_ThanksRegister {
get {
return ResourceManager.GetString("Support_ThanksRegister", resourceCulture);
}
}
/// <summary>
/// Looks up a localized string similar to Username.
/// </summary>
@ -1113,6 +1193,42 @@ public static string Support_UserName {
}
}
/// <summary>
/// Looks up a localized string similar to Please check the validation link in your email and try again..
/// </summary>
public static string Support_ValidationFailureContent {
get {
return ResourceManager.GetString("Support_ValidationFailureContent", resourceCulture);
}
}
/// <summary>
/// Looks up a localized string similar to We were unable to validate your account..
/// </summary>
public static string Support_ValidationFailureTitle {
get {
return ResourceManager.GetString("Support_ValidationFailureTitle", resourceCulture);
}
}
/// <summary>
/// Looks up a localized string similar to You will now be able to log in using your username and password..
/// </summary>
public static string Support_ValidationSuccessContent {
get {
return ResourceManager.GetString("Support_ValidationSuccessContent", resourceCulture);
}
}
/// <summary>
/// Looks up a localized string similar to You have successfully validated your account..
/// </summary>
public static string Support_ValidationSuccessTitle {
get {
return ResourceManager.GetString("Support_ValidationSuccessTitle", resourceCulture);
}
}
/// <summary>
/// Looks up a localized string similar to Week.
/// </summary>

View File

@ -159,6 +159,9 @@
<data name="Common_Admin" xml:space="preserve">
<value>[!!! Âδ₥ïñ !!!]</value>
</data>
<data name="Common_ChangePassword" xml:space="preserve">
<value>[!!! Çλáñϱè ƥáƨƨωôřδ !!!]</value>
</data>
<data name="Common_ContributeOn" xml:space="preserve">
<value>[!!! Çôñƭřïβúƭè ôñ {0} !!!]</value>
</data>
@ -207,6 +210,20 @@
<data name="Common_Twitter" xml:space="preserve">
<value>[!!! Tωïƭƭèř !!!]</value>
</data>
<data name="Email_Registration_Body" xml:space="preserve">
<value>[!!! Tλáñƙ ¥ôú ƒôř řèϱïƨƭèřïñϱ ƭô {0}. ℓôřè !!!]
[!!! !!!]
[!!! Þℓèáƨè Ʋèř ¥ôúř è₥áïℓ áδδřèƨƨ β¥ çℓïçƙïñϱ ƭλè ℓïñƙ βèℓôω, ôř β¥ çôƥ¥ïñϱ áñδ ƥáƨƭïñϱ ïƭ ïñƭô ¥ôúř βřôωƨèř. ℓôřè₥ ïƥƨú₥ δôℓ !!!]
[!!! {1} !!!]
[!!! !!!]
[!!! ̃ ¥ôú δïδ ñôƭ řèϱïƨƭèř ƭô {0}, ¥ôú çáñ ïϱñôřè ƭλïƨ è₥áïℓ. ℓôřè₥ ïƥ !!!]
[!!! !!!]
[!!! Tλáñƙƨ, !!!]
[!!! Tλè {0} Tèá₥. !!!]</value>
</data>
<data name="Email_Registration_Subject" xml:space="preserve">
<value>[!!! {0}: Þℓèáƨè Ʋèř ¥ôúř è₥áïℓ áδδřèƨƨ ℓôřè₥ !!!]</value>
</data>
<data name="Front_About" xml:space="preserve">
<value>[!!! Âβôúƭ !!!]</value>
</data>
@ -405,9 +422,6 @@
<data name="Search_Year" xml:space="preserve">
<value>[!!! Ýèář !!!]</value>
</data>
<data name="Support_AccountValidation" xml:space="preserve">
<value>[!!! ÉƲèř¥ áççôúñƭ ïƨ Ʋáℓïδáƭèδ β¥ áñ Âδ₥ïñïƨƭřáƭôř, ƨô βè ƥáƭïèñƭ áñδ çλèçƙ áϱáïñ ℓáƭèř. ℓôřè₥ ï !!!]</value>
</data>
<data name="Support_AdditionsToBuildFeed" xml:space="preserve">
<value>[!!! Âδδïƭïôñƨ ƭô ßúïδFèèδ !!!]</value>
</data>
@ -426,6 +440,12 @@
<data name="Support_EmailAddress" xml:space="preserve">
<value>[!!! É₥áïℓ áδδřèƨƨ !!!]</value>
</data>
<data name="Support_EmailValidationContent" xml:space="preserve">
<value>[!!! Âñ Ʋáℓïδáƭïôñ ℓïñƙ λáƨ βèèñ ƨèñƭ ƭô ¥ôúř è₥áïℓ áδδřèƨƨ. Þℓèáƨè çℓïçƙ ƭλè ℓïñƙ ïñ ƭλè è₥áïℓ ƭô Ʋèř ƭλè áççôúñƭ. ℓôřè₥ ïƥƨú₥ δôℓô !!!]</value>
</data>
<data name="Support_EmailValidationTitle" xml:space="preserve">
<value>[!!! Þℓèáƨè Ʋèř ¥ôúř è₥áïℓ áδδřèƨƨ ℓôřè !!!]</value>
</data>
<data name="Support_EnterCurrentPassword" xml:space="preserve">
<value>[!!! Éñƭèř çúřřèñƭ ƥáƨƨωôřδ !!!]</value>
</data>
@ -435,6 +455,21 @@
<data name="Support_EnterPassword" xml:space="preserve">
<value>[!!! Éñƭèř ƥáƨƨωôřδ !!!]</value>
</data>
<data name="Support_Error_ChangingPasswordFail" xml:space="preserve">
<value>[!!! Âñ èřřôř ôççúřřèδ ωλèñ áƭƭè₥ƥƭïñϱ ƭô çλáñϱè ¥ôúř ƥáƨƨωôřδ. ℓôřè₥ ïƥ !!!]</value>
</data>
<data name="Support_Error_DuplicateEmail" xml:space="preserve">
<value>[!!! Â úƨèř áççôúñƭ ωïƭλ ƭλïƨ è₥áïℓ áδδřèƨƨ áℓřèáδ¥ èжïƨƭƨ. ℓôřè₥ ï !!!]</value>
</data>
<data name="Support_Error_DuplicateUserName" xml:space="preserve">
<value>[!!! Â úƨèř áççôúñƭ ωïƭλ ƭλïƨ úƨèř ñá₥è áℓřèáδ¥ èжïƨƭƨ. ℓôřè₥ ï !!!]</value>
</data>
<data name="Support_Error_InvalidPassword" xml:space="preserve">
<value>[!!! Tλè ƥáƨƨωôřδ ¥ôú λáƲè èñƭèřèδ ïƨ ïñƲáℓïδ. ℓôřè₥ !!!]</value>
</data>
<data name="Support_Error_UnknownError" xml:space="preserve">
<value>[!!! Âñ úñƙñôωñ èřřôř λáƨ ôççúřřèδ. ℓôřè !!!]</value>
</data>
<data name="Support_HighestVersion" xml:space="preserve">
<value>[!!! Hïϱλèƨƭ Ʋèřƨïôñ !!!]</value>
</data>
@ -462,12 +497,21 @@
<data name="Support_RememberMe" xml:space="preserve">
<value>[!!! Rè₥è₥βèř ₥è !!!]</value>
</data>
<data name="Support_ThanksRegister" xml:space="preserve">
<value>[!!! Tλáñƙ ¥ôú ƒôř řèϱïƨƭèřïñϱ ℓô !!!]</value>
</data>
<data name="Support_UserName" xml:space="preserve">
<value>[!!! Ûƨèřñá₥è !!!]</value>
</data>
<data name="Support_ValidationFailureContent" xml:space="preserve">
<value>[!!! Þℓèáƨè çλèçƙ ƭλè Ʋáℓïδáƭïôñ ℓïñƙ ïñ ¥ôúř è₥áïℓ áñδ ƭř¥ áϱáïñ. ℓôřè₥ ïƥ !!!]</value>
</data>
<data name="Support_ValidationFailureTitle" xml:space="preserve">
<value>[!!! Wè ωèřè úñáβℓè ƭô Ʋáℓïδáƭè ¥ôúř áççôúñƭ. ℓôřè₥ !!!]</value>
</data>
<data name="Support_ValidationSuccessContent" xml:space="preserve">
<value>[!!! Ýôú ωïℓℓ ñôω βè áβℓè ƭô ℓôϱ ïñ úƨïñϱ ¥ôúř úƨèřñá₥è áñδ ƥáƨƨωôřδ. ℓôřè₥ ïƥƨ !!!]</value>
</data>
<data name="Support_ValidationSuccessTitle" xml:space="preserve">
<value>[!!! Ýôú λáƲè ƨúççèƨƨƒúℓℓ¥ Ʋáℓïδáƭèδ ¥ôúř áççôúñƭ. ℓôřè₥ !!!]</value>
</data>
<data name="Support_Week" xml:space="preserve">
<value>[!!! Wèèƙ !!!]</value>
</data>

View File

@ -159,6 +159,9 @@
<data name="Common_Admin" xml:space="preserve">
<value>Admin</value>
</data>
<data name="Common_ChangePassword" xml:space="preserve">
<value>Change password</value>
</data>
<data name="Common_ContributeOn" xml:space="preserve">
<value>Contribute on {0}</value>
</data>
@ -207,6 +210,20 @@
<data name="Common_Twitter" xml:space="preserve">
<value>Twitter</value>
</data>
<data name="Email_Registration_Body" xml:space="preserve">
<value>Thank you for registering to {0}.
Please verify your email address by clicking the link below, or by copying and pasting it into your browser.
{1}
If you did not register to {0}, you can ignore this email.
Thanks,
The {0} Team.</value>
</data>
<data name="Email_Registration_Subject" xml:space="preserve">
<value>{0}: Please verify your email address</value>
</data>
<data name="Front_About" xml:space="preserve">
<value>About</value>
</data>
@ -405,9 +422,6 @@
<data name="Search_Year" xml:space="preserve">
<value>Year</value>
</data>
<data name="Support_AccountValidation" xml:space="preserve">
<value>Every account is validated by an administrator, so be patient and check again later</value>
</data>
<data name="Support_AdditionsToBuildFeed" xml:space="preserve">
<value>Additions to BuildFeed</value>
</data>
@ -426,6 +440,12 @@
<data name="Support_EmailAddress" xml:space="preserve">
<value>Email address</value>
</data>
<data name="Support_EmailValidationContent" xml:space="preserve">
<value>An validation link has been sent to your email address. Please click the link in the email to verify the account.</value>
</data>
<data name="Support_EmailValidationTitle" xml:space="preserve">
<value>Please verify your email address</value>
</data>
<data name="Support_EnterCurrentPassword" xml:space="preserve">
<value>Enter current password</value>
</data>
@ -435,6 +455,21 @@
<data name="Support_EnterPassword" xml:space="preserve">
<value>Enter password</value>
</data>
<data name="Support_Error_ChangingPasswordFail" xml:space="preserve">
<value>An error occurred when attempting to change your password.</value>
</data>
<data name="Support_Error_DuplicateEmail" xml:space="preserve">
<value>A user account with this email address already exists.</value>
</data>
<data name="Support_Error_DuplicateUserName" xml:space="preserve">
<value>A user account with this user name already exists.</value>
</data>
<data name="Support_Error_InvalidPassword" xml:space="preserve">
<value>The password you have entered is invalid.</value>
</data>
<data name="Support_Error_UnknownError" xml:space="preserve">
<value>An unknown error has occurred.</value>
</data>
<data name="Support_HighestVersion" xml:space="preserve">
<value>Highest version</value>
</data>
@ -462,12 +497,21 @@
<data name="Support_RememberMe" xml:space="preserve">
<value>Remember me</value>
</data>
<data name="Support_ThanksRegister" xml:space="preserve">
<value>Thank you for registering</value>
</data>
<data name="Support_UserName" xml:space="preserve">
<value>Username</value>
</data>
<data name="Support_ValidationFailureContent" xml:space="preserve">
<value>Please check the validation link in your email and try again.</value>
</data>
<data name="Support_ValidationFailureTitle" xml:space="preserve">
<value>We were unable to validate your account.</value>
</data>
<data name="Support_ValidationSuccessContent" xml:space="preserve">
<value>You will now be able to log in using your username and password.</value>
</data>
<data name="Support_ValidationSuccessTitle" xml:space="preserve">
<value>You have successfully validated your account.</value>
</data>
<data name="Support_Week" xml:space="preserve">
<value>Week</value>
</data>

View File

@ -46,17 +46,15 @@
<a href="mailto:@mu.Email">@mu.Email</a>
</td>
<td>
@(mu.CreationDate == default(DateTime)
? "Never"
: mu.CreationDate.Humanize())
@mu.CreationDate.Humanize()
</td>
<td>
@(mu.LastLoginDate == default(DateTime)
@(mu.LastLoginDate == DateTime.MinValue
? "Never"
: mu.LastLoginDate.Humanize())
</td>
<td>
@(mu.LastLockoutDate == default(DateTime)
@(mu.LastLockoutDate == DateTime.MinValue
? "Never"
: mu.LastLockoutDate.Humanize())
</td>

View File

@ -73,23 +73,23 @@
<Reference Include="Microsoft.AI.Agent.Intercept, Version=2.0.7.0, Culture=neutral, PublicKeyToken=31bf3856ad364e35, processorArchitecture=MSIL">
<HintPath>..\packages\Microsoft.ApplicationInsights.Agent.Intercept.2.0.7\lib\net45\Microsoft.AI.Agent.Intercept.dll</HintPath>
</Reference>
<Reference Include="Microsoft.AI.DependencyCollector, Version=2.2.0.0, Culture=neutral, PublicKeyToken=31bf3856ad364e35, processorArchitecture=MSIL">
<HintPath>..\packages\Microsoft.ApplicationInsights.DependencyCollector.2.2.0\lib\net45\Microsoft.AI.DependencyCollector.dll</HintPath>
<Reference Include="Microsoft.AI.DependencyCollector, Version=2.3.0.0, Culture=neutral, PublicKeyToken=31bf3856ad364e35, processorArchitecture=MSIL">
<HintPath>..\packages\Microsoft.ApplicationInsights.DependencyCollector.2.3.0\lib\net45\Microsoft.AI.DependencyCollector.dll</HintPath>
</Reference>
<Reference Include="Microsoft.AI.PerfCounterCollector, Version=2.2.0.0, Culture=neutral, PublicKeyToken=31bf3856ad364e35, processorArchitecture=MSIL">
<HintPath>..\packages\Microsoft.ApplicationInsights.PerfCounterCollector.2.2.0\lib\net45\Microsoft.AI.PerfCounterCollector.dll</HintPath>
<Reference Include="Microsoft.AI.PerfCounterCollector, Version=2.3.0.0, Culture=neutral, PublicKeyToken=31bf3856ad364e35, processorArchitecture=MSIL">
<HintPath>..\packages\Microsoft.ApplicationInsights.PerfCounterCollector.2.3.0\lib\net45\Microsoft.AI.PerfCounterCollector.dll</HintPath>
</Reference>
<Reference Include="Microsoft.AI.ServerTelemetryChannel, Version=2.2.0.0, Culture=neutral, PublicKeyToken=31bf3856ad364e35, processorArchitecture=MSIL">
<HintPath>..\packages\Microsoft.ApplicationInsights.WindowsServer.TelemetryChannel.2.2.0\lib\net45\Microsoft.AI.ServerTelemetryChannel.dll</HintPath>
<Reference Include="Microsoft.AI.ServerTelemetryChannel, Version=2.3.0.0, Culture=neutral, PublicKeyToken=31bf3856ad364e35, processorArchitecture=MSIL">
<HintPath>..\packages\Microsoft.ApplicationInsights.WindowsServer.TelemetryChannel.2.3.0\lib\net45\Microsoft.AI.ServerTelemetryChannel.dll</HintPath>
</Reference>
<Reference Include="Microsoft.AI.Web, Version=2.2.0.0, Culture=neutral, PublicKeyToken=31bf3856ad364e35, processorArchitecture=MSIL">
<HintPath>..\packages\Microsoft.ApplicationInsights.Web.2.2.0\lib\net45\Microsoft.AI.Web.dll</HintPath>
<Reference Include="Microsoft.AI.Web, Version=2.3.0.0, Culture=neutral, PublicKeyToken=31bf3856ad364e35, processorArchitecture=MSIL">
<HintPath>..\packages\Microsoft.ApplicationInsights.Web.2.3.0\lib\net45\Microsoft.AI.Web.dll</HintPath>
</Reference>
<Reference Include="Microsoft.AI.WindowsServer, Version=2.2.0.0, Culture=neutral, PublicKeyToken=31bf3856ad364e35, processorArchitecture=MSIL">
<HintPath>..\packages\Microsoft.ApplicationInsights.WindowsServer.2.2.0\lib\net45\Microsoft.AI.WindowsServer.dll</HintPath>
<Reference Include="Microsoft.AI.WindowsServer, Version=2.3.0.0, Culture=neutral, PublicKeyToken=31bf3856ad364e35, processorArchitecture=MSIL">
<HintPath>..\packages\Microsoft.ApplicationInsights.WindowsServer.2.3.0\lib\net45\Microsoft.AI.WindowsServer.dll</HintPath>
</Reference>
<Reference Include="Microsoft.ApplicationInsights, Version=2.2.0.0, Culture=neutral, PublicKeyToken=31bf3856ad364e35, processorArchitecture=MSIL">
<HintPath>..\packages\Microsoft.ApplicationInsights.2.2.0\lib\net46\Microsoft.ApplicationInsights.dll</HintPath>
<Reference Include="Microsoft.ApplicationInsights, Version=2.3.0.0, Culture=neutral, PublicKeyToken=31bf3856ad364e35, processorArchitecture=MSIL">
<HintPath>..\packages\Microsoft.ApplicationInsights.2.3.0\lib\net46\Microsoft.ApplicationInsights.dll</HintPath>
</Reference>
<Reference Include="Microsoft.CodeDom.Providers.DotNetCompilerPlatform, Version=1.0.3.0, Culture=neutral, PublicKeyToken=31bf3856ad364e35, processorArchitecture=MSIL">
<HintPath>..\packages\Microsoft.CodeDom.Providers.DotNetCompilerPlatform.1.0.3\lib\net45\Microsoft.CodeDom.Providers.DotNetCompilerPlatform.dll</HintPath>
@ -109,10 +109,10 @@
<HintPath>..\packages\MongoDB.Driver.Core.2.4.3\lib\net45\MongoDB.Driver.Core.dll</HintPath>
</Reference>
<Reference Include="Newtonsoft.Json, Version=10.0.0.0, Culture=neutral, PublicKeyToken=30ad4fe6b2a6aeed, processorArchitecture=MSIL">
<HintPath>..\packages\Newtonsoft.Json.10.0.1\lib\net45\Newtonsoft.Json.dll</HintPath>
<HintPath>..\packages\Newtonsoft.Json.10.0.2\lib\net45\Newtonsoft.Json.dll</HintPath>
</Reference>
<Reference Include="OneSignal.CSharp.SDK, Version=1.0.0.0, Culture=neutral, processorArchitecture=MSIL">
<HintPath>..\packages\OneSignal.CSharp.SDK.0.9\lib\net45\OneSignal.CSharp.SDK.dll</HintPath>
<HintPath>..\packages\OneSignal.CSharp.SDK.0.10\lib\net45\OneSignal.CSharp.SDK.dll</HintPath>
</Reference>
<Reference Include="RestSharp, Version=105.2.3.0, Culture=neutral, processorArchitecture=MSIL">
<HintPath>..\packages\RestSharp.105.2.3\lib\net46\RestSharp.dll</HintPath>
@ -195,6 +195,7 @@
<Compile Include="Areas\admin\Controllers\metaController.cs" />
<Compile Include="Areas\admin\Models\ViewModel\MetaListing.cs" />
<Compile Include="Code\AiHandleErrorAttribute.cs" />
<Compile Include="Code\Base32Encoding.cs" />
<Compile Include="Code\CustomContentTypeAttribute.cs" />
<Compile Include="Code\DateTimeModelBinder.cs" />
<Compile Include="Code\DisplayHelpers.cs" />
@ -202,11 +203,14 @@
<Compile Include="App_Start\RouteConfig.cs" />
<Compile Include="Areas\admin\adminAreaRegistration.cs" />
<Compile Include="Areas\admin\Controllers\usersController.cs" />
<Compile Include="Code\EmailManager\EmailManager.cs" />
<Compile Include="Code\EmailManager\RegistrationEmail.cs" />
<Compile Include="Code\MvcIntrinsics.cs" />
<Compile Include="Code\OneSignalHelper.cs" />
<Compile Include="Code\Options\Locale.cs" />
<Compile Include="Code\Options\Theme.cs" />
<Compile Include="Code\OutputCachePushAttribute.cs" />
<Compile Include="Controllers\AccountController.cs" />
<Compile Include="Controllers\ApiController.cs" />
<Compile Include="Controllers\BaseController.cs" />
<Compile Include="Controllers\FrontController.cs" />
@ -391,10 +395,10 @@
<Content Include="Views\shared\_default.cshtml" />
<Content Include="Views\front\EditBuild.cshtml" />
<Content Include="Views\shared\DisplayTemplates\Enumeration.cshtml" />
<Content Include="Views\support\register.cshtml" />
<Content Include="Views\support\login.cshtml" />
<Content Include="Views\support\password.cshtml" />
<Content Include="Views\support\thanks_register.cshtml" />
<Content Include="Views\account\register.cshtml" />
<Content Include="Views\account\login.cshtml" />
<Content Include="Views\account\password.cshtml" />
<Content Include="Views\account\registerthanks.cshtml" />
<Content Include="Views\support\rss.cshtml" />
<Content Include="Views\support\sitemap.cshtml" />
<Content Include="smtp.config">
@ -416,6 +420,8 @@
<Content Include="Views\front\AddBulk.cshtml" />
<Content Include="Scripts\trumbowyg\plugins\colors\ui\sass\trumbowyg.colors.scss" />
<Content Include="Scripts\trumbowyg\ui\sass\trumbowyg.scss" />
<Content Include="Views\account\validate-success.cshtml" />
<Content Include="Views\account\validate-failure.cshtml" />
</ItemGroup>
<ItemGroup />
<ItemGroup>

View File

@ -0,0 +1,2 @@
<wpf:ResourceDictionary xml:space="preserve" xmlns:x="http://schemas.microsoft.com/winfx/2006/xaml" xmlns:s="clr-namespace:System;assembly=mscorlib" xmlns:ss="urn:shemas-jetbrains-com:settings-storage-xaml" xmlns:wpf="http://schemas.microsoft.com/winfx/2006/xaml/presentation">
<s:Boolean x:Key="/Default/CodeInspection/NamespaceProvider/NamespaceFoldersToSkip/=code_005Cemailmanager/@EntryIndexedValue">True</s:Boolean></wpf:ResourceDictionary>

View File

@ -0,0 +1,133 @@
using System;
namespace BuildFeed.Code
{
/* Based off http://stackoverflow.com/a/7135008 */
public class Base32Encoding
{
public static byte[] ToBytes(string input)
{
if (string.IsNullOrEmpty(input))
{
throw new ArgumentNullException(nameof(input));
}
input = input.TrimEnd('='); //remove padding characters
int byteCount = input.Length * 5 / 8; //this must be TRUNCATED
var returnArray = new byte[byteCount];
byte curByte = 0,
bitsRemaining = 8;
int arrayIndex = 0;
foreach (char c in input)
{
int cValue = CharToValue(c);
int mask;
if (bitsRemaining > 5)
{
mask = cValue << (bitsRemaining - 5);
curByte = (byte)(curByte | mask);
bitsRemaining -= 5;
}
else
{
mask = cValue >> (5 - bitsRemaining);
curByte = (byte)(curByte | mask);
returnArray[arrayIndex++] = curByte;
curByte = (byte)(cValue << (3 + bitsRemaining));
bitsRemaining += 3;
}
}
//if we didn't end with a full byte
if (arrayIndex != byteCount)
{
returnArray[arrayIndex] = curByte;
}
return returnArray;
}
public static string ToString(byte[] input)
{
if (input == null || input.Length == 0)
{
throw new ArgumentNullException(nameof(input));
}
int charCount = (int)Math.Ceiling(input.Length / 5d) * 8;
var returnArray = new char[charCount];
byte nextChar = 0,
bitsRemaining = 5;
int arrayIndex = 0;
foreach (byte b in input)
{
nextChar = (byte)(nextChar | (b >> (8 - bitsRemaining)));
returnArray[arrayIndex++] = ValueToChar(nextChar);
if (bitsRemaining < 4)
{
nextChar = (byte)((b >> (3 - bitsRemaining)) & 31);
returnArray[arrayIndex++] = ValueToChar(nextChar);
bitsRemaining += 5;
}
bitsRemaining -= 3;
nextChar = (byte)((b << bitsRemaining) & 31);
}
//if we didn't end with a full char
if (arrayIndex != charCount)
{
returnArray[arrayIndex++] = ValueToChar(nextChar);
while (arrayIndex != charCount)
returnArray[arrayIndex++] = '='; //padding
}
return new string(returnArray);
}
private static int CharToValue(char c)
{
int value = c;
//65-90 == uppercase letters
if (value < 91 && value > 64)
{
return value - 65;
}
//50-55 == numbers 2-7
if (value < 56 && value > 49)
{
return value - 24;
}
//97-122 == lowercase letters
if (value < 123 && value > 96)
{
return value - 97;
}
throw new ArgumentException("Character is not a Base32 character.", nameof(c));
}
private static char ValueToChar(byte b)
{
if (b < 26)
{
return (char)(b + 65);
}
if (b < 32)
{
return (char)(b + 24);
}
throw new ArgumentException("Byte is not a value Base32 value.", nameof(b));
}
}
}

View File

@ -0,0 +1,7 @@
namespace BuildFeed.Code
{
public static partial class EmailManager
{
private const string EMAIL_FROM = "thomas@buildfeed.net";
}
}

View File

@ -0,0 +1,24 @@
using System.Net.Mail;
using System.Threading.Tasks;
using System.Web.Security;
using BuildFeed.Local;
namespace BuildFeed.Code
{
public static partial class EmailManager
{
public static async Task SendRegistrationEmail(MembershipUser mu, string validationLink)
{
using (MailMessage mm = new MailMessage(EMAIL_FROM, mu.Email))
{
mm.Subject = string.Format(VariantTerms.Email_Registration_Subject, InvariantTerms.SiteName);
mm.Body = string.Format(VariantTerms.Email_Registration_Body, InvariantTerms.SiteName, validationLink);
using (SmtpClient sc = new SmtpClient())
{
await sc.SendMailAsync(mm);
}
}
}
}
}

View File

@ -0,0 +1,156 @@
using System;
using System.Net;
using System.Threading.Tasks;
using System.Web;
using System.Web.Mvc;
using System.Web.Security;
using BuildFeed.Code;
using BuildFeed.Local;
using BuildFeed.Model.View;
using MongoAuth;
namespace BuildFeed.Controllers
{
public class AccountController : BaseController
{
[Route("login/")]
public ActionResult Login() => View();
[HttpPost]
[ValidateAntiForgeryToken]
[Route("login/")]
public ActionResult Login(LoginUser ru)
{
if (ModelState.IsValid)
{
bool isAuthenticated = Membership.ValidateUser(ru.UserName, ru.Password);
if (isAuthenticated)
{
int expiryLength = ru.RememberMe
? 129600
: 60;
FormsAuthenticationTicket ticket = new FormsAuthenticationTicket(ru.UserName, true, expiryLength);
string encryptedTicket = FormsAuthentication.Encrypt(ticket);
HttpCookie cookieTicket = new HttpCookie(FormsAuthentication.FormsCookieName, encryptedTicket)
{
Expires = DateTime.Now.AddMinutes(expiryLength),
Path = FormsAuthentication.FormsCookiePath
};
Response.Cookies.Add(cookieTicket);
string returnUrl = string.IsNullOrEmpty(Request.QueryString["ReturnUrl"])
? "/"
: Request.QueryString["ReturnUrl"];
return Redirect(returnUrl);
}
}
ViewData["ErrorMessage"] = "The username and password are not valid.";
return View(ru);
}
[Authorize]
[Route("password/")]
public ActionResult Password() => View();
[HttpPost]
[ValidateAntiForgeryToken]
[Authorize]
[Route("password/")]
public ActionResult Password(ChangePassword cp)
{
if (ModelState.IsValid)
{
MembershipUser user = Membership.GetUser();
if (user != null)
{
bool success = user.ChangePassword(cp.OldPassword, cp.NewPassword);
if (success)
{
return Redirect("/");
}
}
}
ViewData["ErrorMessage"] = VariantTerms.Support_Error_ChangingPasswordFail;
return View(cp);
}
[Route("logout/")]
public ActionResult Logout()
{
FormsAuthentication.SignOut();
return Redirect("/");
}
[Route("register/")]
public ActionResult Register() => View();
[HttpPost]
[ValidateAntiForgeryToken]
[Route("register/")]
public async Task<ActionResult> Register(RegistrationUser ru)
{
if (ModelState.IsValid)
{
MembershipCreateStatus status;
MembershipUser mu = Membership.CreateUser(ru.UserName, ru.Password, ru.EmailAddress, "{IGNORE}", "{IGNORE}", false, out status);
switch (status)
{
case MembershipCreateStatus.Success:
{
Roles.AddUserToRole(mu.UserName, "Users");
MongoMembershipProvider provider = (MongoMembershipProvider)Membership.Provider;
Guid id = (Guid)mu.ProviderUserKey;
string hash = (await provider.GenerateValidationHash(id)).ToLower();
string fullUrl = Request.Url?.GetLeftPart(UriPartial.Authority) + Url.Action("Validate",
"Account",
new
{
id,
hash
});
await EmailManager.SendRegistrationEmail(mu, fullUrl);
return RedirectToAction(nameof(RegisterThanks));
}
case MembershipCreateStatus.InvalidPassword:
ViewData["ErrorMessage"] = VariantTerms.Support_Error_InvalidPassword;
break;
case MembershipCreateStatus.DuplicateEmail:
ViewData["ErrorMessage"] = VariantTerms.Support_Error_DuplicateEmail;
break;
case MembershipCreateStatus.DuplicateUserName:
ViewData["ErrorMessage"] = VariantTerms.Support_Error_DuplicateUserName;
break;
default:
ViewData["ErrorMessage"] = VariantTerms.Support_Error_UnknownError;
break;
}
}
return View(ru);
}
[Route("register/thanks/")]
public ActionResult RegisterThanks() => View();
[Route("validate/{id:guid}/{hash}/")]
public async Task<ActionResult> Validate(Guid id, string hash)
{
MongoMembershipProvider provider = (MongoMembershipProvider)Membership.Provider;
bool success = await provider.ValidateUserFromHash(id, hash);
return View(success
? "validate-success"
: "validate-failure");
}
}
}

View File

@ -12,6 +12,7 @@
using BuildFeed.Model.View;
using OneSignal.CSharp.SDK;
#pragma warning disable SG0016 // Controller method is vulnerable to CSRF - Not relevant for API
namespace BuildFeed.Controllers
{
public class ApiController : System.Web.Http.ApiController
@ -70,7 +71,7 @@ public async Task<bool> AddWin10Builds(NewBuildPost apiModel)
{
return false;
}
if (Membership.ValidateUser(apiModel.Username, apiModel.Password))
if (Membership.ValidateUser(apiModel.Username, apiModel.Password) && (Roles.IsUserInRole(apiModel.Username, "Editors") || Roles.IsUserInRole(apiModel.Username, "Administrators")))
{
IEnumerable<Build> builds = apiModel.NewBuilds.Select(nb => new Build
{

View File

@ -378,7 +378,7 @@ public async Task<ActionResult> ViewVersionPage(uint major, uint minor, int page
}
[Route("add/")]
[Authorize]
[Authorize(Roles = "Administrators,Editors")]
public ActionResult AddBuild()
{
Build b = new Build
@ -389,7 +389,8 @@ public ActionResult AddBuild()
}
[Route("add/")]
[Authorize]
[ValidateAntiForgeryToken]
[Authorize(Roles = "Administrators,Editors")]
[HttpPost]
public async Task<ActionResult> AddBuild(Build build)
{
@ -429,14 +430,15 @@ public async Task<ActionResult> AddBuild(Build build)
}
[Route("bulk/")]
[Authorize]
[Authorize(Roles = "Administrators")]
public ActionResult AddBulk()
{
return View();
}
[Route("bulk/")]
[Authorize]
[ValidateAntiForgeryToken]
[Authorize(Roles = "Administrators")]
[HttpPost]
public async Task<ActionResult> AddBulk(FormCollection values)
{
@ -509,7 +511,7 @@ public async Task<ActionResult> AddBulk(FormCollection values)
}
[Route("edit/{id}/")]
[Authorize]
[Authorize(Roles = "Administrators,Editors")]
public async Task<ActionResult> EditBuild(Guid id)
{
Build b = await _bModel.SelectById(id);
@ -517,7 +519,8 @@ public async Task<ActionResult> EditBuild(Guid id)
}
[Route("edit/{id}/")]
[Authorize]
[ValidateAntiForgeryToken]
[Authorize(Roles = "Administrators,Editors")]
[HttpPost]
public async Task<ActionResult> EditBuild(Guid id, Build build)
{

View File

@ -2,10 +2,8 @@
using System.Collections.Generic;
using System.Linq;
using System.Threading.Tasks;
using System.Web;
using System.Web.Mvc;
using System.Web.Routing;
using System.Web.Security;
using System.Xml.Linq;
using BuildFeed.Code;
using BuildFeed.Local;
@ -23,116 +21,6 @@ public SupportController()
_bModel = new BuildRepository();
}
[Route("login/")]
public ActionResult Login() => View();
[HttpPost]
[Route("login/")]
public ActionResult Login(LoginUser ru)
{
if (ModelState.IsValid)
{
bool isAuthenticated = Membership.ValidateUser(ru.UserName, ru.Password);
if (isAuthenticated)
{
int expiryLength = ru.RememberMe
? 129600
: 60;
FormsAuthenticationTicket ticket = new FormsAuthenticationTicket(ru.UserName, true, expiryLength);
string encryptedTicket = FormsAuthentication.Encrypt(ticket);
HttpCookie cookieTicket = new HttpCookie(FormsAuthentication.FormsCookieName, encryptedTicket)
{
Expires = DateTime.Now.AddMinutes(expiryLength),
Path = FormsAuthentication.FormsCookiePath
};
Response.Cookies.Add(cookieTicket);
string returnUrl = string.IsNullOrEmpty(Request.QueryString["ReturnUrl"])
? "/"
: Request.QueryString["ReturnUrl"];
return Redirect(returnUrl);
}
}
ViewData["ErrorMessage"] = "The username and password are not valid.";
return View(ru);
}
[Authorize]
[Route("password/")]
public ActionResult Password() => View();
[HttpPost]
[Authorize]
[Route("password/")]
public ActionResult Password(ChangePassword cp)
{
if (ModelState.IsValid)
{
MembershipUser user = Membership.GetUser();
if (user != null)
{
bool success = user.ChangePassword(cp.OldPassword, cp.NewPassword);
if (success)
{
return Redirect("/");
}
}
}
ViewData["ErrorMessage"] = "There was an error changing your password.";
return View(cp);
}
[Route("logout/")]
public ActionResult Logout()
{
FormsAuthentication.SignOut();
return Redirect("/");
}
[Route("register/")]
public ActionResult Register() => View();
[HttpPost]
[Route("register/")]
public ActionResult Register(RegistrationUser ru)
{
if (ModelState.IsValid)
{
MembershipCreateStatus status;
Membership.CreateUser(ru.UserName, ru.Password, ru.EmailAddress, "THIS WILL BE IGNORED", "I WILL BE IGNORED", false, out status);
switch (status)
{
case MembershipCreateStatus.Success:
return RedirectToAction("thanks_register");
case MembershipCreateStatus.InvalidPassword:
ViewData["ErrorMessage"] = "The password is invalid.";
break;
case MembershipCreateStatus.DuplicateEmail:
ViewData["ErrorMessage"] = "A user account with this email address already exists.";
break;
case MembershipCreateStatus.DuplicateUserName:
ViewData["ErrorMessage"] = "A user account with this user name already exists.";
break;
default:
ViewData["ErrorMessage"] = "Unspecified error.";
break;
}
}
return View(ru);
}
[Route("register/thanks/")]
public ActionResult thanks_register() => View();
[Route("rss")]
public async Task<ActionResult> Rss()
{

View File

@ -4,6 +4,7 @@
using System.Web;
using System.Web.Mvc;
using System.Web.Routing;
using System.Web.Security;
using BuildFeed.Code;
using BuildFeed.Code.Options;
using BuildFeed.Model;
@ -30,6 +31,10 @@ protected void Application_Start()
ModelBinders.Binders.Add(typeof(DateTime), db);
ModelBinders.Binders.Add(typeof(DateTime?), db);
Roles.CreateRole("Administrators");
Roles.CreateRole("Editors");
Roles.CreateRole("Users");
MongoConfig.SetupIndexes();
}

View File

@ -1,4 +1,5 @@
@model BuildFeed.Model.View.LoginUser
@using BuildFeed.Controllers
@model BuildFeed.Model.View.LoginUser
@{
ViewBag.Title = $"{VariantTerms.Support_Login} | {InvariantTerms.SiteName}";
@ -57,9 +58,9 @@
<div class="form-group">
<div>
<input type="submit" value="@VariantTerms.Support_Login" class="btn btn-primary" /> &ensp;
@Html.ActionLink(VariantTerms.Support_Register, "register", new
@Html.ActionLink(VariantTerms.Support_Register, nameof(AccountController.Register), new
{
controller = "Support"
controller = "Account"
}, new
{
@class = "btn btn-default"

View File

@ -0,0 +1,6 @@
@{
ViewBag.Title = $"{VariantTerms.Support_EmailValidationTitle} | {InvariantTerms.SiteName}";
}
<h1>@VariantTerms.Support_EmailValidationTitle</h1>
<p>@VariantTerms.Support_EmailValidationContent</p>

View File

@ -0,0 +1,6 @@
@{
ViewBag.Title = $"{VariantTerms.Support_ValidationFailureTitle} | {InvariantTerms.SiteName}";
}
<h1>@VariantTerms.Support_ValidationFailureTitle</h1>
<p>@VariantTerms.Support_ValidationFailureContent</p>

View File

@ -0,0 +1,6 @@
@{
ViewBag.Title = $"{VariantTerms.Support_ValidationSuccessTitle} | {InvariantTerms.SiteName}";
}
<h1>@VariantTerms.Support_ValidationSuccessTitle</h1>
<p>@VariantTerms.Support_ValidationSuccessContent</p>

View File

@ -143,7 +143,7 @@
<div class="addthis_sharing_toolbox"></div>
<br />
@if (User.Identity.IsAuthenticated)
@if (Roles.IsUserInRole("Editors") || Roles.IsUserInRole("Administrators"))
{
<h3>@VariantTerms.Front_EditorActions</h3>
<p class="build-details-flex-value">

View File

@ -49,7 +49,7 @@
<i class="fa fa-lock fa-fw"></i> @VariantTerms.Front_Private</span>
</p>
}
@if (User.Identity.IsAuthenticated)
@if (Roles.IsUserInRole("Editors") || Roles.IsUserInRole("Administrators"))
{
<p>
<a href="@Url.Action(nameof(FrontController.EditBuild), new

View File

@ -78,7 +78,7 @@
<i class="fa fa-lock fa-fw"></i> @VariantTerms.Front_Private</span>
</p>
}
@if (User.Identity.IsAuthenticated)
@if (Roles.IsUserInRole("Editors") || Roles.IsUserInRole("Administrators"))
{
<p>
<a href="@Url.Action(nameof(FrontController.EditBuild), new

View File

@ -75,7 +75,7 @@
<i class="fa fa-lock fa-fw"></i> @VariantTerms.Front_Private</span>
</p>
}
@if (User.Identity.IsAuthenticated)
@if (Roles.IsUserInRole("Editors") || Roles.IsUserInRole("Administrators"))
{
<p>
<a href="@Url.Action(nameof(FrontController.EditBuild), new

View File

@ -74,7 +74,7 @@
<i class="fa fa-lock fa-fw"></i> @VariantTerms.Front_Private</span>
</p>
}
@if (User.Identity.IsAuthenticated)
@if (Roles.IsUserInRole("Editors") || Roles.IsUserInRole("Administrators"))
{
<p>
<a href="@Url.Action(nameof(FrontController.EditBuild), new

View File

@ -84,7 +84,7 @@
<i class="fa fa-lock fa-fw"></i> @VariantTerms.Front_Private</span>
</p>
}
@if (User.Identity.IsAuthenticated)
@if (Roles.IsUserInRole("Editors") || Roles.IsUserInRole("Administrators"))
{
<p>
<a href="@Url.Action(nameof(FrontController.EditBuild), new

View File

@ -60,7 +60,7 @@
<script type="text/javascript">
var appInsights = window.appInsights || function (config) {
function i(config) { t[config] = function () { var i = arguments; t.queue.push(function () { t[config].apply(t, i) }) } } var t = { config: config }, u = document, e = window, o = "script", s = "AuthenticatedUserContext", h = "start", c = "stop", l = "Track", a = l + "Event", v = l + "Page", y = u.createElement(o), r, f; y.src = config.url || "https://az416426.vo.msecnd.net/scripts/a/ai.0.js"; u.getElementsByTagName(o)[0].parentNode.appendChild(y); try { t.cookie = u.cookie } catch (p) { } for (t.queue = [], t.version = "1.0", r = ["Event", "Exception", "Metric", "PageView", "Trace", "Dependency"]; r.length;)i("track" + r.pop()); return i("set" + s), i("clear" + s), i(h + a), i(c + a), i(h + v), i(c + v), i("flush"), config.disableExceptionTracking || (r = "onerror", i("_" + r), f = e[r], e[r] = function (config, i, u, e, o) { var s = f && f(config, i, u, e, o); return s !== !0 && t["_" + r](config, i, u, e, o), s }), t
} ({
}({
instrumentationKey: "4632419f-7a2f-4ab5-8374-34384b650f42"
});
@ -69,7 +69,7 @@
</script>
</head>
<body>
<script>
<script>
(function (i, s, o, g, r, a, m) {
i['GoogleAnalyticsObject'] = r; i[r] = i[r] || function () {
(i[r].q = i[r].q || []).push(arguments);
@ -81,8 +81,8 @@
ga('require', 'displayfeatures');
ga('require', 'linkid', 'linkid.js');
ga('send', 'pageview');
</script>
<header id="page-header">
</script>
<header id="page-header">
<div class="container">
<h1>
@Html.ActionLink(InvariantTerms.SiteName, nameof(FrontController.Index), new
@ -92,8 +92,8 @@
})
</h1>
</div>
</header>
<nav id="page-navigation" role="navigation">
</header>
<nav id="page-navigation" role="navigation">
<div class="container">
<button id="page-navigation-toggle">
<i class="fa fa-bars"></i>&ensp;@VariantTerms.Common_ToggleNavigation
@ -102,9 +102,9 @@
@if (!User.Identity.IsAuthenticated)
{
<li>
<a href="@Url.Action(nameof(SupportController.Login), new
<a href="@Url.Action(nameof(AccountController.Login), new
{
controller = "Support",
controller = "Account",
area = ""
})" title="@VariantTerms.Common_LogIn">
<i class="fa fa-fw fa-user"></i> @VariantTerms.Common_LogIn
@ -125,6 +125,8 @@
</a>
</li>
}
if (Roles.IsUserInRole("Administrators") || Roles.IsUserInRole("Editors"))
{
<li>
<a href="@Url.Action(nameof(FrontController.AddBuild), new
{
@ -143,15 +145,25 @@
<i class="fa fa-fw fa-database"></i> @VariantTerms.Common_AddBulk
</a>
</li>
}
<li>
<a href="@Url.Action(nameof(SupportController.Logout), new
<a href="@Url.Action(nameof(AccountController.Logout), new
{
controller = "Support",
controller = "Account",
area = ""
})" title="@VariantTerms.Common_LogOut">
<i class="fa fa-fw fa-user"></i> @VariantTerms.Common_LogOut
</a>
</li>
<li>
<a href="@Url.Action(nameof(AccountController.Password), new
{
controller = "Account",
area = ""
})" title="@VariantTerms.Common_ChangePassword">
<i class="fa fa-fw fa-key"></i> @VariantTerms.Common_ChangePassword
</a>
</li>
}
<li>
<a href="#" id="page-navigation-search" title="@VariantTerms.Search_Title">
@ -197,7 +209,8 @@
<a href="#" data-lang="@locale.LocaleId" dir="@(locale.Info.TextInfo.IsRightToLeft
? "rtl"
: "ltr")">
@locale.DisplayName</a>
@locale.DisplayName
</a>
</li>
}
<li></li>
@ -212,13 +225,13 @@
</li>
</ul>
</div>
</nav>
<article id="page-content">
</nav>
<article id="page-content">
<div class="container">
@RenderBody()
</div>
</article>
<footer id="page-footer">
</article>
<footer id="page-footer">
<div class="container">
<div class="footer-flex">
<div class="footer-flex-item">
@ -228,7 +241,8 @@
controller = "Support",
area = ""
})">
@VariantTerms.Common_Sitemap</a>
@VariantTerms.Common_Sitemap
</a>
</p>
<p>
<i class="fa fa-language"></i>&ensp;
@ -255,9 +269,9 @@
</div>
</div>
</div>
</footer>
</footer>
<div id="modal-search-overlay">
<div id="modal-search-overlay">
<div id="modal-search">
<h3>@VariantTerms.Search_BuildFeed</h3>
<div id="modal-search-box">
@ -268,18 +282,18 @@
</div>
<div id="modal-search-result"></div>
</div>
</div>
<div id="menu-open-overlay"></div>
</div>
<div id="menu-open-overlay"></div>
<script type="text/javascript" src="https://cdnjs.cloudflare.com/ajax/libs/jsrender/1.0.0-rc.70/jsrender.min.js" integrity="sha256-3UBtL0tzgKVuJU8ZZiWLXEWGEjXEr6Z023rpauMnBUE=" crossorigin="anonymous"></script>
<script type="text/javascript" src="/res/ts/bfs.js"></script>
@RenderSection("scripts", false)
<script id="result-template" type="text/x-jsrender">
<script type="text/javascript" src="https://cdnjs.cloudflare.com/ajax/libs/jsrender/1.0.0-rc.70/jsrender.min.js" integrity="sha256-3UBtL0tzgKVuJU8ZZiWLXEWGEjXEr6Z023rpauMnBUE=" crossorigin="anonymous"></script>
<script type="text/javascript" src="/res/ts/bfs.js"></script>
@RenderSection("scripts", false)
<script id="result-template" type="text/x-jsrender">
<a href="{{:Url}}" class="search-result-item" title="{{:Title}}">
<h4 class="search-result-heading no-wrapping">{{:Label}}</h4>
<p class="search-result-text">{{:Group}}</p>
</a>
</script>
<script type="text/javascript" src="//s7.addthis.com/js/300/addthis_widget.js#pubid=ra-5431719a661cbfd0" async="async"></script>
</script>
<script type="text/javascript" src="//s7.addthis.com/js/300/addthis_widget.js#pubid=ra-5431719a661cbfd0" async="async"></script>
</body>
</html>

View File

@ -1,6 +0,0 @@
@{
ViewBag.Title = $"{VariantTerms.Support_ThanksRegister} | {InvariantTerms.SiteName}";
}
<h1>@VariantTerms.Support_ThanksRegister</h1>
<p>@VariantTerms.Support_AccountValidation</p>

View File

@ -47,13 +47,13 @@
<package id="jquery.TypeScript.DefinitelyTyped" version="3.1.2" targetFramework="net462" />
<package id="jQuery.Validation" version="1.16.0" targetFramework="net462" />
<package id="jsrender.TypeScript.DefinitelyTyped" version="0.1.8" targetFramework="net461" />
<package id="Microsoft.ApplicationInsights" version="2.2.0" targetFramework="net462" />
<package id="Microsoft.ApplicationInsights" version="2.3.0" targetFramework="net462" />
<package id="Microsoft.ApplicationInsights.Agent.Intercept" version="2.0.7" targetFramework="net462" />
<package id="Microsoft.ApplicationInsights.DependencyCollector" version="2.2.0" targetFramework="net462" />
<package id="Microsoft.ApplicationInsights.PerfCounterCollector" version="2.2.0" targetFramework="net462" />
<package id="Microsoft.ApplicationInsights.Web" version="2.2.0" targetFramework="net462" />
<package id="Microsoft.ApplicationInsights.WindowsServer" version="2.2.0" targetFramework="net462" />
<package id="Microsoft.ApplicationInsights.WindowsServer.TelemetryChannel" version="2.2.0" targetFramework="net462" />
<package id="Microsoft.ApplicationInsights.DependencyCollector" version="2.3.0" targetFramework="net462" />
<package id="Microsoft.ApplicationInsights.PerfCounterCollector" version="2.3.0" targetFramework="net462" />
<package id="Microsoft.ApplicationInsights.Web" version="2.3.0" targetFramework="net462" />
<package id="Microsoft.ApplicationInsights.WindowsServer" version="2.3.0" targetFramework="net462" />
<package id="Microsoft.ApplicationInsights.WindowsServer.TelemetryChannel" version="2.3.0" targetFramework="net462" />
<package id="Microsoft.AspNet.Mvc" version="5.2.3" targetFramework="net462" />
<package id="Microsoft.AspNet.Razor" version="3.2.3" targetFramework="net462" />
<package id="Microsoft.AspNet.WebApi" version="5.2.3" targetFramework="net462" />
@ -69,8 +69,8 @@
<package id="MongoDB.Bson" version="2.4.3" targetFramework="net462" />
<package id="MongoDB.Driver" version="2.4.3" targetFramework="net462" />
<package id="MongoDB.Driver.Core" version="2.4.3" targetFramework="net462" />
<package id="Newtonsoft.Json" version="10.0.1" targetFramework="net462" />
<package id="OneSignal.CSharp.SDK" version="0.9" targetFramework="net462" />
<package id="Newtonsoft.Json" version="10.0.2" targetFramework="net462" />
<package id="OneSignal.CSharp.SDK" version="0.10" targetFramework="net462" />
<package id="RestSharp" version="105.2.3" targetFramework="net462" />
<package id="System.Collections" version="4.3.0" targetFramework="net462" />
<package id="System.Linq" version="4.3.0" targetFramework="net462" />

File diff suppressed because one or more lines are too long

File diff suppressed because one or more lines are too long

View File

@ -87,7 +87,8 @@ table
word-wrap: break-word;
}
.field-validation-error
.field-validation-error,
.text-danger
{
display: block;
margin: #{(1em / 3)} 0;
@ -145,7 +146,7 @@ nav#page-navigation
#page-navigation-links
{
margin: 0 -15px;
margin: 0 -15px #{(1px / 3)}; /* bottom margin fixes Chrome in 4k */
padding: 0;
text-align: right;

View File

@ -1,2 +1,2 @@
"use strict";var BuildFeed;!function(e){function t(e){e.preventDefault();var t=this;t.nextElementSibling.classList.toggle("open")}function n(e){e.preventDefault();var t=this;t.parentElement.classList.toggle("open");var n=document.getElementById("menu-open-overlay");n.classList.add("open")}function a(e){e.preventDefault();for(var t=document.getElementsByClassName("dropdown-parent"),n=0;n<t.length;n++)t[n].classList.remove("open");var a=document.getElementById("menu-open-overlay");a.classList.remove("open")}function o(e){e.preventDefault();var t=this;document.cookie="bf_theme="+t.dataset.theme+"; expires=Fri, 31 Dec 9999 23:59:59 GMT; path=/",location.reload(!0)}function r(e){e.preventDefault();var t=this;document.cookie="bf_lang="+t.dataset.lang+"; expires=Fri, 31 Dec 9999 23:59:59 GMT; path=/",location.reload(!0)}function d(e){e.preventDefault();var t=document.getElementById("modal-search-overlay");t.classList.add("open")}function l(e){e.preventDefault();var t=document.getElementById("modal-search-overlay");t.classList.remove("open")}function c(e){e.preventDefault(),e.stopPropagation()}function i(e){var t=document.getElementById("modal-search-result");t.innerHTML="","undefined"!=typeof p&&clearTimeout(p),"undefined"!=typeof v&&v.readyState!==XMLHttpRequest.DONE&&v.abort(),p=setInterval(s,200)}function s(){"undefined"!=typeof p&&clearTimeout(p);var e=document.getElementById("modal-search-input");v=new XMLHttpRequest,v.onreadystatechange=u,v.open("GET","/api/GetSearchResult/"+e.value+"/",!0),v.setRequestHeader("accept","application/json"),v.send(null)}function u(e){if(v.readyState===XMLHttpRequest.DONE&&200===v.status){var t=document.getElementById("modal-search-result"),n=document.getElementById("result-template"),a=jsrender.templates(n.innerHTML),o=a.render(JSON.parse(v.responseText));t.innerHTML=o;for(var r=t.getElementsByTagName("a"),d=0;d<r.length;d++)r[d].addEventListener("click",function(e){e.preventDefault();var t=document.getElementById("modal-search-input");ga("send","pageview","/api/GetSearchResult/"+t.value+"/"),location.assign(e.currentTarget.href)})}}function m(e){for(var s=document.getElementsByClassName("dropdown-parent"),u=0;u<s.length;u++)for(var m=0;m<s[u].childNodes.length;m++){var v=s[u].childNodes[m];"A"===v.nodeName&&v.addEventListener("click",n)}var p=document.getElementById("menu-open-overlay");p.addEventListener("click",a);for(var g=document.getElementById("settings-theme-menu").getElementsByTagName("a"),u=0;u<g.length;u++)g[u].addEventListener("click",o);for(var f=document.getElementById("settings-lang-menu").getElementsByTagName("a"),u=0;u<f.length;u++)f[u].addEventListener("click",r);var h=document.getElementById("page-navigation-toggle");h.addEventListener("click",t);var E=document.getElementById("page-navigation-search");E.addEventListener("click",d);var y=document.getElementById("modal-search-overlay");y.addEventListener("click",l);var B=document.getElementById("modal-search");B.addEventListener("click",c);var L=document.getElementById("modal-search-input");L.addEventListener("keyup",i)}var v,p;e.MobileMenuToggle=t,e.DropdownClick=n,e.CloseDropdowns=a,e.SwitchTheme=o,e.SwitchLanguage=r,e.OpenSearch=d,e.CloseSearch=l,e.StopClick=c,e.InitiateSearch=i,e.SendSearch=s,e.CompleteSearch=u,e.BuildFeedSetup=m}(BuildFeed||(BuildFeed={})),window.addEventListener("load",BuildFeed.BuildFeedSetup);
"use strict";var BuildFeed;!function(e){function t(e){e.preventDefault(),this.nextElementSibling.classList.toggle("open")}function n(e){e.preventDefault(),this.parentElement.classList.toggle("open"),document.getElementById("menu-open-overlay").classList.add("open")}function a(e){e.preventDefault();for(var t=document.getElementsByClassName("dropdown-parent"),n=0;n<t.length;n++)t[n].classList.remove("open");document.getElementById("menu-open-overlay").classList.remove("open")}function o(e){e.preventDefault();var t=this;document.cookie="bf_theme="+t.dataset.theme+"; expires=Fri, 31 Dec 9999 23:59:59 GMT; path=/",location.reload(!0)}function l(e){e.preventDefault();var t=this;document.cookie="bf_lang="+t.dataset.lang+"; expires=Fri, 31 Dec 9999 23:59:59 GMT; path=/",location.reload(!0)}function d(e){e.preventDefault(),document.getElementById("modal-search-overlay").classList.add("open")}function r(e){e.preventDefault(),document.getElementById("modal-search-overlay").classList.remove("open")}function c(e){e.preventDefault(),e.stopPropagation()}function i(e){document.getElementById("modal-search-result").innerHTML="",void 0!==p&&clearTimeout(p),void 0!==g&&g.readyState!==XMLHttpRequest.DONE&&g.abort(),p=setInterval(s,200)}function s(){void 0!==p&&clearTimeout(p);var e=document.getElementById("modal-search-input");g=new XMLHttpRequest,g.onreadystatechange=u,g.open("GET","/api/GetSearchResult/"+e.value+"/",!0),g.setRequestHeader("accept","application/json"),g.send(null)}function u(e){if(g.readyState===XMLHttpRequest.DONE&&200===g.status){var t=document.getElementById("modal-search-result"),n=document.getElementById("result-template"),a=jsrender.templates(n.innerHTML),o=a.render(JSON.parse(g.responseText));t.innerHTML=o;for(var l=t.getElementsByTagName("a"),d=0;d<l.length;d++)l[d].addEventListener("click",function(e){e.preventDefault();var t=document.getElementById("modal-search-input");ga("send","pageview","/api/GetSearchResult/"+t.value+"/"),location.assign(e.currentTarget.href)})}}function m(e){for(var s=document.getElementsByClassName("dropdown-parent"),u=0;u<s.length;u++)for(var m=0;m<s[u].childNodes.length;m++){var g=s[u].childNodes[m];"A"===g.nodeName&&g.addEventListener("click",n)}document.getElementById("menu-open-overlay").addEventListener("click",a);for(var p=document.getElementById("settings-theme-menu").getElementsByTagName("a"),u=0;u<p.length;u++)p[u].addEventListener("click",o);for(var v=document.getElementById("settings-lang-menu").getElementsByTagName("a"),u=0;u<v.length;u++)v[u].addEventListener("click",l);document.getElementById("page-navigation-toggle").addEventListener("click",t),document.getElementById("page-navigation-search").addEventListener("click",d),document.getElementById("modal-search-overlay").addEventListener("click",r),document.getElementById("modal-search").addEventListener("click",c),document.getElementById("modal-search-input").addEventListener("keyup",i)}var g,p;e.MobileMenuToggle=t,e.DropdownClick=n,e.CloseDropdowns=a,e.SwitchTheme=o,e.SwitchLanguage=l,e.OpenSearch=d,e.CloseSearch=r,e.StopClick=c,e.InitiateSearch=i,e.SendSearch=s,e.CompleteSearch=u,e.BuildFeedSetup=m}(BuildFeed||(BuildFeed={})),window.addEventListener("load",BuildFeed.BuildFeedSetup);
//# sourceMappingURL=bfs.js.map

File diff suppressed because one or more lines are too long