Files
2024-06-18 17:29:05 -04:00

271 lines
8.0 KiB
PHP

<?php
/**
* Handles logic for honeypot service.
*
* @package Gravity_Forms\Gravity_Forms\Honeypot
*/
namespace Gravity_Forms\Gravity_Forms\Honeypot;
/**
* Class GF_Honeypot_Handler
*
* @since 2.7
*
* Provides functionality for handling honeypot spam prevention services.
*/
class GF_Honeypot_Handler {
/**
* Target of the gform_entry_is_spam filter. Checks entry for honeypot validation and returns true or false depending on the result.
*
* @since 2.7
*
* @param bool $is_spam Variable being filtered. True for spam, false for non-spam.
* @param array $form Current form object.
*
* @return bool Returns true if honeypot validation fails. False otherwise.
*/
public function handle_entry_is_spam( $is_spam, $form ) {
// If already marked as spam, don't change it.
if ( $is_spam ) {
return true;
}
// Bypass honeypot validation if disabled.
if ( ! $this->is_honeypot_enabled( $form ) ) {
return false;
}
$is_spam = ! $this->validate_honeypot( $form );
// Setting filter that flagged entry as spam so that an entry note is created.
if ( $is_spam ) {
\GFCommon::set_spam_filter( $form['id'], __( 'Honeypot Spam Filter', 'gravityforms' ), __( 'Failed Honeypot Validation.', 'gravityforms' ) );
}
return $is_spam;
}
/**
* Target of the gform_abort_submission_with_confirmation filter. Aborts form submission early with a confirmation when honeypot fails and it is configured not to create an entry.
*
* @since 2.7
*
* @param bool $do_abort Variable being filtered. True to abort submission, false to continue.
* @param array $form Current form object.
*
* @return bool Returns true to abort form submission early and display confirmation. Returns false to let submission continue.
*/
public function handle_abort_submission( $do_abort, $form ) {
// If already marked to abort early, let it abort early.
if ( $do_abort ) {
return true;
}
// Do not abort submission if Honeypot should be disabled or if honeypot action is set to create an entry.
if ( ! $this->is_honeypot_enabled( $form ) || rgar( $form, 'honeypotAction' ) == 'spam' ) {
return false;
}
$do_abort = ! $this->validate_honeypot( $form );
\GFCommon::log_debug( __METHOD__ . '(): Result from Honeypot: ' . json_encode( $do_abort ) );
return $do_abort;
}
/**
* Target of the gform_after_submission. Clears the cached results.
*
* @since 2.7
*
* @param array $entry Current entry object.
* @param array $form Current form object.
*/
public function handle_after_submission( $entry, $form ) {
\GFCache::delete( "honeypot_{$form['id']}" );
}
/**
* Adds the honeypot field to the form if honeypot is enabled.
*
* @since 2.7
*
* @param array $form Current form object.
*
* @return array Returns a form object with the new honeypot field appended to the fields array.
*/
public function maybe_add_honeypot_field( $form ) {
if ( rgar( $form, 'enableHoneypot' ) ) {
$form['fields'][] = $this->get_honeypot_field( $form );
}
return $form;
}
/**
* Validates the submission against the honeypot field.
*
* @since 2.7
*
* @param array $form The current form object.
*
* @return bool True if form passes the honeypot validation (i.e. Not spam). False if honeypot validation fails (i.e. spam)
*/
public function validate_honeypot( $form ) {
// If validation has already been computed for this form, no need to validate it again.
$cache_key = "honeypot_{$form['id']}";
if ( \GFCache::get( $cache_key ) !== false ) {
return (bool) \GFCache::get( $cache_key );
}
$honeypot_id = $this->get_honeypot_field_id( $form );
$pass_server_side_honeypot = rgempty( "input_{$honeypot_id}" );
\GFCommon::log_debug( __METHOD__ . '(): Is honeypot input empty? ' . json_encode( $pass_server_side_honeypot ) );
// Bypass JS field hash validation on GFAPI submissions.
if ( $this->is_api_submission() ) {
$pass_js_honeypot = true;
\GFCommon::log_debug( __METHOD__ . '(): Submission initiated by GFAPI. Honeypot JS field hash validation bypassed.' );
} else {
$pass_js_honeypot = $this->is_valid_version_hash( rgpost( 'version_hash' ) );
\GFCommon::log_debug( __METHOD__ . '(): Is version_hash input valid? ' . json_encode( $pass_js_honeypot ) );
}
$is_success = $pass_server_side_honeypot && $pass_js_honeypot;
\GFCommon::log_debug( __METHOD__ . '(): Are both inputs valid? ' . json_encode( $is_success ) );
\GFCache::set( $cache_key, (int) $is_success );
return $is_success;
}
/**
* Returns the ID of the honeypot field.
*
* @since 2.7
*
* @param array $form Current form object.
*
* @return int Returns the id of the honeypot field.
*/
public function get_honeypot_field_id( $form ) {
if ( empty( $form['fields'] ) ) {
return 0;
}
// Look for honeypot field in the form.
$honeypot_field = \GFFormsModel::get_fields_by_type( $form, array( 'honeypot' ) );
if ( count( $honeypot_field ) > 0 ) {
return $honeypot_field[0]->id;
}
// If no honeypot field in the form, return the largest field ID + 1.
return \GFFormsModel::get_next_field_id( $form['fields'] );
}
/**
* Creates the honeypot field object for the given form.
*
* @since 2.7
*
* @param array $form The form the honeypot field is to be created for.
*
* @return GF_Field Returns the honeypot field.
*/
private function get_honeypot_field( $form ) {
$labels = $this->get_honeypot_labels();
$field_data = array(
'type' => 'honeypot',
'label' => $labels[ rand( 0, count( $labels ) - 1 ) ],
'id' => \GFFormsModel::get_next_field_id( $form['fields'] ),
'cssClass' => 'gform_validation_container',
'description' => __( 'This field is for validation purposes and should be left unchanged.', 'gravityforms' ),
'formId' => absint( $form['id'] ),
);
return \GF_Fields::create( $field_data );
}
/**
* Returns an array of possible labels to be used for the Honeypot field.
*
* @since 2.7
*
* @return array Returns an array of possible labels
*/
private function get_honeypot_labels() {
$honeypot_labels = array( 'Name', 'Email', 'Phone', 'Comments' );
/**
* Allow the honeypot field labels to be overridden.
*
* @since 2.0.7.16
*
* @param array $honeypot_labels The honeypot field labels.
*/
return apply_filters( 'gform_honeypot_labels_pre_render', $honeypot_labels );
}
/**
* Validates a version hash.
*
* @since 2.7
*
* @param string $hash The version hash to be validated.
*
* @return bool Returns true if the hash is validated against the current and previous version of Gravity Forms
*/
public function is_valid_version_hash( $hash ) {
// Allow password to validate on current version and previous version.
$allowed_hashes = array( wp_hash( \GFForms::$version ) );
$previous_version = get_option( 'gf_previous_db_version' );
if ( ! empty( $previous_version ) ) {
$allowed_hashes[] = wp_hash( $previous_version );
}
return in_array( $hash, $allowed_hashes );
}
/**
* Determines if Honeypot should be enabled for this form submission.
*
* @since 2.7
*
* @param array $form The current form object.
*
* @return bool True if honeypot should be enabled. False otherwise.
*/
public function is_honeypot_enabled( $form ) {
// Honeypot should be disabled if ANY of the following is true:
// 1- honeypot is not enabled by this form in form settings.
// 2- the form is submitted from preview.
// 3- the form is submitted from the WP dashboard.
$is_wp_dashboard = is_admin() && ! ( defined( 'DOING_AJAX' ) && DOING_AJAX );
$is_disabled = ! rgar( $form, 'enableHoneypot' ) || \GFCommon::is_preview() || $is_wp_dashboard;
return ! $is_disabled;
}
/**
* Determines if the current form submission was initiated via GFAPI.
*
* @since 2.7
*
* @return bool True if the current form submission was initiated via GFAPI. False otherwise.
*/
public function is_api_submission() {
return \GFFormDisplay::$submission_initiated_by == \GFFormDisplay::SUBMISSION_INITIATED_BY_API;
}
}