Orchard/src/Orchard.Web/Modules/Orchard.AntiSpam/Drivers/ReCaptchaPartDriver.cs
Matteo Piovanelli 104d18aa06 ReCaptchaPartDriver always returns a shape (#8292)
* We changed when ReCaptchaPartDriver verifies whether it's being executed on the Admin side, moving the test from the driver's execution to the shape's, so that the ShapeType is available to other features and services that process the driver's output (e.g. for placement)

* store the first validation of recaptcha to avoid failing if the driver runs again in a single request
2019-12-05 12:06:53 -08:00

145 lines
6.0 KiB
C#

using System;
using System.Globalization;
using System.IO;
using System.Net;
using System.Text;
using System.Web;
using Orchard.AntiSpam.Models;
using Orchard.AntiSpam.ViewModels;
using Orchard.ContentManagement;
using Orchard.ContentManagement.Drivers;
using Orchard.Localization;
using Orchard.Logging;
using Orchard.UI.Admin;
using Orchard.UI.Notify;
using Orchard.Services;
using System.Collections.Generic;
namespace Orchard.AntiSpam.Drivers {
public class ReCaptchaPartDriver : ContentPartDriver<ReCaptchaPart> {
private readonly INotifier _notifier;
private readonly IJsonConverter _jsonConverter;
private readonly IWorkContextAccessor _workContextAccessor;
private const string ReCaptchaSecureUrl = "https://www.google.com/recaptcha/api/siteverify";
public ReCaptchaPartDriver(
INotifier notifier,
IJsonConverter jsonConverter,
IWorkContextAccessor workContextAccessor) {
_notifier = notifier;
_jsonConverter = jsonConverter;
_workContextAccessor = workContextAccessor;
T = NullLocalizer.Instance;
Logger = NullLogger.Instance;
}
public Localizer T { get; set; }
public ILogger Logger { get; set; }
protected override DriverResult Editor(ReCaptchaPart part, dynamic shapeHelper) {
// we want to be returning a shape even when it should display nothing, because
// other features may need the Shape's type, or some other of its properties
return ContentShape("Parts_ReCaptcha_Fields", () => {
var workContext = _workContextAccessor.GetContext();
// don't display the part in the admin
if (AdminFilter.IsApplied(workContext.HttpContext.Request.RequestContext)) {
return null;
}
var settings = workContext.CurrentSite.As<ReCaptchaSettingsPart>();
if (settings.TrustAuthenticatedUsers && workContext.CurrentUser != null) {
return null;
}
var viewModel = new ReCaptchaPartEditViewModel {
PublicKey = settings.PublicKey
};
return shapeHelper.EditorTemplate(TemplateName: "Parts.ReCaptcha.Fields", Model: viewModel, Prefix: Prefix);
});
}
protected override DriverResult Editor(ReCaptchaPart part, IUpdateModel updater, dynamic shapeHelper) {
var workContext = _workContextAccessor.GetContext();
var settings = workContext.CurrentSite.As<ReCaptchaSettingsPart>();
// don't display the part in the admin
if (AdminFilter.IsApplied(workContext.HttpContext.Request.RequestContext)) {
return null;
}
if (settings.TrustAuthenticatedUsers && workContext.CurrentUser != null) {
return null;
}
var context = workContext.HttpContext;
try {
var result = ValidateRequest(//ExecuteValidateRequest(
settings.PrivateKey,
context.Request.ServerVariables["REMOTE_ADDR"],
context.Request.Form["g-recaptcha-response"]
);
ReCaptchaPartResponseModel responseModel = _jsonConverter.Deserialize<ReCaptchaPartResponseModel>(result);
if (!responseModel.Success) {
foreach (var errorCode in responseModel.ErrorCodes) {
if(errorCode == "missing-input-response") {
updater.AddModelError("", T("Please prove that you are not a bot."));
_notifier.Error(T("Please prove that you are not a bot."));
}
else {
Logger.Information("An error occurred while submitting a reCaptcha: " + errorCode);
updater.AddModelError("", T("An error occurred while submitting a reCaptcha."));
_notifier.Error(T("An error occurred while submitting a reCaptcha."));
}
}
}
}
catch (Exception e) {
Logger.Error(e, "An unexcepted error occurred while submitting a reCaptcha.");
updater.AddModelError("", T("There was an error while validating the Captcha."));
_notifier.Error(T("There was an error while validating the Captcha."));
}
return Editor(part, shapeHelper);
}
// temporarily save <postData, response> pairs, to prevent sending the same exact request
// more than once in a single Request.
private Dictionary<string, string> ValidationResponse;
private string ValidateRequest(string privateKey, string remoteip, string response) {
if (ValidationResponse == null) {
ValidationResponse = new Dictionary<string, string>();
}
var postData = string.Format(CultureInfo.InvariantCulture,
"secret={0}&response={1}&remoteip={2}",
privateKey,
response,
remoteip
);
if (!ValidationResponse.ContainsKey(postData)) {
ValidationResponse.Add(postData, ExecuteValidateRequest(postData));
}
return ValidationResponse[postData];
}
private static string ExecuteValidateRequest(string postData) {
WebRequest request = WebRequest.Create(ReCaptchaSecureUrl + "?" + postData);
request.Method = "GET";
request.Timeout = 5000; //milliseconds
request.ContentType = "application/x-www-form-urlencoded";
using (WebResponse webResponse = request.GetResponse()) {
using (var reader = new StreamReader(webResponse.GetResponseStream())) {
return reader.ReadToEnd();
}
}
}
}
}