mirror of
https://github.com/WordPress/five-for-the-future.git
synced 2025-04-16 16:43:42 +03:00

`FILTER_UNSAFE_RAW` doesn't strip HTML tags like `FILTER_SANITIZE_STRING` did, but some of these are being run through `sanitize_text_field` as well, and the others aren't being output or saved to the database. See https://stackoverflow.com/a/69207369/450127
587 lines
16 KiB
PHP
Executable file
587 lines
16 KiB
PHP
Executable file
<?php
|
|
/**
|
|
* This file handles the operations related to registering and handling pledge meta values for the CPT.
|
|
*/
|
|
|
|
namespace WordPressDotOrg\FiveForTheFuture\PledgeMeta;
|
|
|
|
use WordPressDotOrg\FiveForTheFuture;
|
|
use WordPressDotOrg\FiveForTheFuture\{ Auth, Contributor, Email, Pledge, PledgeForm, XProfile };
|
|
use WP_Post, WP_Error;
|
|
|
|
defined( 'WPINC' ) || die();
|
|
|
|
const META_PREFIX = FiveForTheFuture\PREFIX . '_';
|
|
|
|
add_action( 'init', __NAMESPACE__ . '\register_pledge_meta' );
|
|
add_action( 'init', __NAMESPACE__ . '\schedule_cron_jobs' );
|
|
add_action( 'admin_init', __NAMESPACE__ . '\add_meta_boxes' );
|
|
add_action( 'save_post', __NAMESPACE__ . '\save_pledge', 10, 2 );
|
|
add_action( 'admin_enqueue_scripts', __NAMESPACE__ . '\enqueue_assets' );
|
|
add_action( 'wp_enqueue_scripts', __NAMESPACE__ . '\enqueue_assets' );
|
|
add_action( 'transition_post_status', __NAMESPACE__ . '\maybe_update_single_cached_pledge_data', 10, 3 );
|
|
add_action( 'update_all_cached_pledge_data', __NAMESPACE__. '\update_all_cached_pledge_data' );
|
|
|
|
// Both hooks must be used because `updated` doesn't fire if the post meta didn't previously exist.
|
|
add_action( 'updated_postmeta', __NAMESPACE__ . '\update_generated_meta', 10, 4 );
|
|
add_action( 'added_post_meta', __NAMESPACE__ . '\update_generated_meta', 10, 4 );
|
|
// Email (re-)confirmation is only hooked into updates, otherwise it sends duplicate emails.
|
|
add_action( 'updated_postmeta', __NAMESPACE__ . '\trigger_confirmation_email', 10, 4 );
|
|
|
|
/**
|
|
* Define pledge meta fields and their properties.
|
|
*
|
|
* @param string $subset Optional. The part of the config to return: 'user_input', 'generated', or 'all'.
|
|
*
|
|
* @return array
|
|
*/
|
|
function get_pledge_meta_config( $subset = 'all' ) {
|
|
$user_input = array(
|
|
'org-description' => array(
|
|
'single' => true,
|
|
'sanitize_callback' => __NAMESPACE__ . '\sanitize_description',
|
|
'show_in_rest' => true,
|
|
'context' => array( 'create', 'update' ),
|
|
'php_filter' => FILTER_UNSAFE_RAW,
|
|
),
|
|
'org-name' => array(
|
|
'single' => true,
|
|
'sanitize_callback' => 'sanitize_text_field',
|
|
'show_in_rest' => true,
|
|
'context' => array( 'create', 'update' ),
|
|
'php_filter' => FILTER_UNSAFE_RAW,
|
|
),
|
|
'org-url' => array(
|
|
'single' => true,
|
|
'sanitize_callback' => 'esc_url_raw',
|
|
'show_in_rest' => true,
|
|
'context' => array( 'create', 'update' ),
|
|
'php_filter' => FILTER_VALIDATE_URL,
|
|
),
|
|
'org-pledge-email' => array(
|
|
'single' => true,
|
|
'sanitize_callback' => 'sanitize_email',
|
|
'show_in_rest' => false,
|
|
'context' => array( 'create' ),
|
|
'php_filter' => FILTER_VALIDATE_EMAIL,
|
|
),
|
|
);
|
|
|
|
$generated = array(
|
|
'org-domain' => array(
|
|
'single' => true,
|
|
'sanitize_callback' => 'sanitize_text_field',
|
|
'show_in_rest' => false,
|
|
),
|
|
'pledge-email-confirmed' => array(
|
|
'single' => true,
|
|
'sanitize_callback' => 'wp_validate_boolean',
|
|
'show_in_rest' => false,
|
|
),
|
|
'pledge-confirmed-contributors' => array(
|
|
'single' => true,
|
|
'sanitize_callback' => 'absint',
|
|
'show_in_rest' => false,
|
|
),
|
|
'pledge-total-hours' => array(
|
|
'single' => true,
|
|
'sanitize_callback' => 'absint',
|
|
'show_in_rest' => false,
|
|
),
|
|
);
|
|
|
|
switch ( $subset ) {
|
|
case 'user_input':
|
|
$return = $user_input;
|
|
break;
|
|
case 'generated':
|
|
$return = $generated;
|
|
break;
|
|
default:
|
|
$return = array_merge( $user_input, $generated );
|
|
break;
|
|
}
|
|
|
|
return $return;
|
|
}
|
|
|
|
/**
|
|
* Sanitize description fields.
|
|
*
|
|
* @param string $insecure
|
|
*
|
|
* @return string
|
|
*/
|
|
function sanitize_description( $insecure ) {
|
|
$secure = wp_kses_data( $insecure );
|
|
$secure = wp_unslash( wp_rel_nofollow( $secure ) );
|
|
|
|
return $secure;
|
|
}
|
|
|
|
/**
|
|
* Register post meta keys for the custom post type.
|
|
*
|
|
* @return void
|
|
*/
|
|
function register_pledge_meta() {
|
|
$meta = get_pledge_meta_config();
|
|
|
|
foreach ( $meta as $key => $args ) {
|
|
$meta_key = META_PREFIX . $key;
|
|
|
|
register_post_meta( Pledge\CPT_ID, $meta_key, $args );
|
|
}
|
|
}
|
|
|
|
/**
|
|
* Schedule cron jobs.
|
|
*
|
|
* This needs to run on the `init` action, because Cavalcade isn't fully loaded before that, and events
|
|
* wouldn't be scheduled.
|
|
*
|
|
* @see https://dotorg.trac.wordpress.org/changeset/15351/
|
|
*/
|
|
function schedule_cron_jobs() {
|
|
if ( ! wp_next_scheduled( 'update_all_cached_pledge_data' ) ) {
|
|
wp_schedule_event( time(), 'hourly', 'update_all_cached_pledge_data' );
|
|
}
|
|
}
|
|
|
|
/**
|
|
* Regularly update the cached data for all pledges.
|
|
*
|
|
* Outside of this cron job, it's only updated when a contributor post changes status, but it's possible for
|
|
* a contributor to edit their profile's # of hours at any time. If we didn't regularly check and update it,
|
|
* then it could be permanently out of date.
|
|
*/
|
|
function update_all_cached_pledge_data() {
|
|
$pledges = get_posts( array(
|
|
'post_type' => Pledge\CPT_ID,
|
|
'post_status' => 'publish',
|
|
'numberposts' => 1000,
|
|
) );
|
|
|
|
foreach ( $pledges as $pledge ) {
|
|
update_single_cached_pledge_data( $pledge->ID );
|
|
}
|
|
}
|
|
|
|
/**
|
|
* Adds meta boxes for the custom post type.
|
|
*
|
|
* @return void
|
|
*/
|
|
function add_meta_boxes() {
|
|
add_meta_box(
|
|
'pledge-email',
|
|
__( 'Pledge Email', 'wporg-5ftf' ),
|
|
__NAMESPACE__ . '\render_meta_boxes',
|
|
Pledge\CPT_ID,
|
|
'normal',
|
|
'high'
|
|
);
|
|
|
|
add_meta_box(
|
|
'org-info',
|
|
__( 'Organization Information', 'wporg-5ftf' ),
|
|
__NAMESPACE__ . '\render_meta_boxes',
|
|
Pledge\CPT_ID,
|
|
'normal',
|
|
'high'
|
|
);
|
|
|
|
add_meta_box(
|
|
'pledge-contributors',
|
|
__( 'Contributors', 'wporg-5ftf' ),
|
|
__NAMESPACE__ . '\render_meta_boxes',
|
|
Pledge\CPT_ID,
|
|
'normal',
|
|
'high'
|
|
);
|
|
}
|
|
|
|
/**
|
|
* Builds the markup for all meta boxes
|
|
*
|
|
* @param WP_Post $pledge
|
|
* @param array $box
|
|
*/
|
|
function render_meta_boxes( $pledge, $box ) {
|
|
$readonly = (
|
|
! current_user_can( 'edit_page', $pledge->ID ) ||
|
|
Pledge\DEACTIVE_STATUS === $pledge->post_status
|
|
);
|
|
$is_manage = true;
|
|
$pledge_id = $pledge->ID;
|
|
|
|
$data = array();
|
|
foreach ( get_pledge_meta_config() as $key => $config ) {
|
|
$data[ $key ] = get_post_meta( $pledge->ID, META_PREFIX . $key, $config['single'] );
|
|
}
|
|
|
|
$contributors = Contributor\get_pledge_contributors_data( $pledge->ID );
|
|
|
|
echo '<div class="pledge-form">';
|
|
|
|
switch ( $box['id'] ) {
|
|
case 'pledge-email':
|
|
require FiveForTheFuture\get_views_path() . 'inputs-pledge-org-email.php';
|
|
break;
|
|
|
|
case 'org-info':
|
|
require FiveForTheFuture\get_views_path() . 'inputs-pledge-org-info.php';
|
|
break;
|
|
|
|
case 'pledge-contributors':
|
|
require FiveForTheFuture\get_views_path() . 'manage-contributors.php';
|
|
break;
|
|
}
|
|
|
|
echo '</div>';
|
|
}
|
|
|
|
/**
|
|
* Save the pledge data.
|
|
*
|
|
* This only fires when the pledge post itself is created or updated.
|
|
*
|
|
* @param int $pledge_id
|
|
* @param WP_Post $pledge
|
|
*/
|
|
function save_pledge( $pledge_id, $pledge ) {
|
|
$get_action = filter_input( INPUT_GET, 'action' );
|
|
$post_action = filter_input( INPUT_POST, 'action' );
|
|
$ignored_actions = array( 'trash', 'untrash', 'restore' );
|
|
$context = ( 'editpost' === $post_action ) ? 'update' : 'create';
|
|
|
|
/*
|
|
* This is only intended to run when the front end form and wp-admin forms are submitted, not when posts are
|
|
* programmatically updated.
|
|
*/
|
|
if ( ! in_array( $post_action, array( 'Submit Pledge', 'editpost' ), true ) ) {
|
|
return;
|
|
}
|
|
|
|
if ( $get_action && in_array( $get_action, $ignored_actions, true ) ) {
|
|
return;
|
|
}
|
|
|
|
if ( ! $pledge instanceof WP_Post || Pledge\CPT_ID !== $pledge->post_type ) {
|
|
return;
|
|
}
|
|
|
|
// if ( ! current_user_can( 'edit_pledge', $pledge_id ) ) {} -- todo re-enable once setup cap mapping or whatever.
|
|
|
|
if ( ( defined( 'DOING_AUTOSAVE' ) && DOING_AUTOSAVE ) || 'auto-draft' === $pledge->post_status ) {
|
|
return;
|
|
}
|
|
|
|
$submitted_meta = PledgeForm\get_form_submission();
|
|
|
|
if ( is_wp_error( has_required_pledge_meta( $submitted_meta, $context ) ) ) {
|
|
return;
|
|
}
|
|
|
|
save_pledge_meta( $pledge_id, $submitted_meta );
|
|
|
|
// Fired if the "Resend Confirmation" button was clicked in wp-admin.
|
|
if ( filter_input( INPUT_POST, 'resend-pledge-confirmation' ) ) {
|
|
Email\send_pledge_confirmation_email(
|
|
filter_input( INPUT_GET, 'resend-pledge-id', FILTER_VALIDATE_INT ),
|
|
get_page_by_path( 'manage-pledge' )->ID
|
|
);
|
|
}
|
|
}
|
|
|
|
/**
|
|
* Save the pledge's meta fields.
|
|
*
|
|
* @param int $pledge_id
|
|
* @param array $new_values
|
|
*
|
|
* @return void
|
|
*/
|
|
function save_pledge_meta( $pledge_id, $new_values ) {
|
|
$config = get_pledge_meta_config();
|
|
|
|
foreach ( $new_values as $key => $value ) {
|
|
// A null value can happen if the submission form did not have a given field.
|
|
if ( is_null( $value ) ) {
|
|
continue;
|
|
}
|
|
if ( array_key_exists( $key, $config ) ) {
|
|
$meta_key = META_PREFIX . $key;
|
|
|
|
// Since the sanitize callback is called during this function, it could still end up
|
|
// saving an empty value to the database.
|
|
update_post_meta( $pledge_id, $meta_key, $value );
|
|
}
|
|
}
|
|
}
|
|
|
|
/**
|
|
* Updated some generated meta values based on changes in user input meta values.
|
|
*
|
|
* This is hooked to the `updated_{$meta_type}_meta` & `added_{$meta_type}_meta` actions, which fires if a
|
|
* submitted post meta value is new or different from the previous value. Thus here we assume the values of
|
|
* specific meta keys are changed when they come through this function.
|
|
*
|
|
* @param int $meta_id
|
|
* @param int $object_id
|
|
* @param string $meta_key
|
|
* @param mixed $_meta_value
|
|
*
|
|
* @return void
|
|
*/
|
|
function update_generated_meta( $meta_id, $object_id, $meta_key, $_meta_value ) {
|
|
$post_type = get_post_type( $object_id );
|
|
|
|
if ( Pledge\CPT_ID !== $post_type ) {
|
|
return;
|
|
}
|
|
|
|
switch ( $meta_key ) {
|
|
case META_PREFIX . 'org-name':
|
|
if ( 'updated_postmeta' === current_action() ) {
|
|
wp_update_post( array(
|
|
'ID' => $object_id,
|
|
'post_title' => $_meta_value,
|
|
) );
|
|
}
|
|
break;
|
|
|
|
case META_PREFIX . 'org-url':
|
|
$domain = get_normalized_domain_from_url( $_meta_value );
|
|
update_post_meta( $object_id, META_PREFIX . 'org-domain', $domain );
|
|
break;
|
|
}
|
|
}
|
|
|
|
/**
|
|
* Trigger the "Confirm your Email" email for pledge owners.
|
|
*
|
|
* See above for hook information, but this function is only hooked into `updated_{$meta_type}_meta`, so it does
|
|
* not run when the pledge is created, only when it's updated.
|
|
*
|
|
* @param int $meta_id
|
|
* @param int $object_id
|
|
* @param string $meta_key
|
|
* @param mixed $_meta_value
|
|
*
|
|
* @return void
|
|
*/
|
|
function trigger_confirmation_email( $meta_id, $object_id, $meta_key, $_meta_value ) {
|
|
$post_type = get_post_type( $object_id );
|
|
|
|
if ( Pledge\CPT_ID !== $post_type ) {
|
|
return;
|
|
}
|
|
|
|
if ( META_PREFIX . 'org-pledge-email' === $meta_key ) {
|
|
$landing_page = get_page_by_path( 'manage-pledge' )->ID;
|
|
Email\send_pledge_confirmation_email( $object_id, $landing_page );
|
|
delete_post_meta( $object_id, META_PREFIX . 'pledge-email-confirmed' );
|
|
// Flip pledge back to pending.
|
|
wp_update_post( array(
|
|
'ID' => $object_id,
|
|
'post_status' => 'pending',
|
|
) );
|
|
}
|
|
}
|
|
|
|
/**
|
|
* Update the cached pledge data when a contributor post changes statuses.
|
|
*
|
|
* Note that contributor posts should always be trashed instead of deleted completely when a contributor is
|
|
* removed from a pledge.
|
|
*
|
|
* @param string $new_status
|
|
* @param string $old_status
|
|
* @param WP_Post $post
|
|
*
|
|
* @return void
|
|
*/
|
|
function maybe_update_single_cached_pledge_data( $new_status, $old_status, WP_Post $post ) {
|
|
if ( Contributor\CPT_ID !== get_post_type( $post ) ) {
|
|
return;
|
|
}
|
|
|
|
if ( $new_status === $old_status ) {
|
|
return;
|
|
}
|
|
|
|
$pledge = get_post( $post->post_parent );
|
|
|
|
if ( $pledge instanceof WP_Post ) {
|
|
update_single_cached_pledge_data( $pledge->ID );
|
|
}
|
|
}
|
|
|
|
/**
|
|
* Update the cached data for the given pledge.
|
|
*
|
|
* This is saved so that it can be easily queried against, and also to make stats calculations easier.
|
|
*
|
|
* @param int $pledge_id
|
|
*/
|
|
function update_single_cached_pledge_data( $pledge_id ) {
|
|
$pledge_data = XProfile\get_aggregate_contributor_data_for_pledge( $pledge_id );
|
|
|
|
update_post_meta( $pledge_id, META_PREFIX . 'pledge-confirmed-contributors', $pledge_data['contributors'] );
|
|
update_post_meta( $pledge_id, META_PREFIX . 'pledge-total-hours', $pledge_data['hours'] );
|
|
}
|
|
|
|
/**
|
|
* Check that an array contains values for all required keys.
|
|
*
|
|
* @param array $submission Form submission data.
|
|
* @param string $context Whether this is a new pledge (`create`) or an edit to an existing one (`update`).
|
|
*
|
|
* @return true|WP_Error True if all required values are present. Otherwise WP_Error.
|
|
*/
|
|
function has_required_pledge_meta( array $submission, $context ) {
|
|
$error = new WP_Error();
|
|
|
|
$meta_config = get_pledge_meta_config( 'user_input' );
|
|
$required = array_keys( $meta_config );
|
|
|
|
foreach ( $required as $key ) {
|
|
if ( ! in_array( $context, $meta_config[ $key ]['context'], true ) ) {
|
|
continue;
|
|
}
|
|
|
|
if ( ! isset( $submission[ $key ] ) || is_null( $submission[ $key ] ) ) {
|
|
$error->add(
|
|
'required_field_empty',
|
|
sprintf(
|
|
__( 'The <code>%s</code> field does not have a value.', 'wporg-5ftf' ),
|
|
sanitize_key( $key )
|
|
)
|
|
);
|
|
} elseif ( false === $submission[ $key ] ) {
|
|
$error->add(
|
|
'required_field_invalid',
|
|
sprintf(
|
|
__( 'The <code>%s</code> field has an invalid value.', 'wporg-5ftf' ),
|
|
sanitize_key( $key )
|
|
)
|
|
);
|
|
}
|
|
}
|
|
|
|
if ( ! empty( $error->get_error_messages() ) ) {
|
|
return $error;
|
|
}
|
|
|
|
return true;
|
|
}
|
|
|
|
/**
|
|
* Get the metadata for a given pledge, or a default set if no pledge is provided.
|
|
*
|
|
* @param int $pledge_id Pledge to fetch data from.
|
|
* @param string $subset Optional. The part of the config to return: 'user_input', 'generated', or 'all'.
|
|
*
|
|
* @return array Pledge data
|
|
*/
|
|
function get_pledge_meta( $pledge_id = 0, $subset = '' ) {
|
|
// Get existing pledge, if it exists.
|
|
$pledge = get_post( $pledge_id );
|
|
|
|
$keys = get_pledge_meta_config( $subset );
|
|
$meta = array();
|
|
|
|
// Get POST'd submission, if it exists.
|
|
$submission = PledgeForm\get_form_submission();
|
|
if ( isset( $submission['empty_post'] ) && $submission['empty_post'] ) {
|
|
$submission = array();
|
|
}
|
|
|
|
foreach ( $keys as $key => $config ) {
|
|
if ( isset( $submission[ $key ] ) ) {
|
|
$meta[ $key ] = $submission[ $key ];
|
|
} elseif ( $pledge instanceof WP_Post ) {
|
|
$meta_key = META_PREFIX . $key;
|
|
$meta[ $key ] = get_post_meta( $pledge->ID, $meta_key, true );
|
|
} else {
|
|
$meta[ $key ] = $config['default'] ?: '';
|
|
}
|
|
}
|
|
|
|
return $meta;
|
|
}
|
|
|
|
/**
|
|
* Isolate the domain from a given URL and remove the `www.` if necessary.
|
|
*
|
|
* @param string $url
|
|
*
|
|
* @return string
|
|
*/
|
|
function get_normalized_domain_from_url( $url ) {
|
|
$domain = wp_parse_url( $url, PHP_URL_HOST );
|
|
$domain = preg_replace( '#^www\.#', '', $domain );
|
|
|
|
return $domain;
|
|
}
|
|
|
|
/**
|
|
* Enqueue CSS file for admin page.
|
|
*
|
|
* @return void
|
|
*/
|
|
function enqueue_assets() {
|
|
wp_register_style(
|
|
'5ftf-admin',
|
|
plugins_url( 'assets/css/admin.css', __DIR__ ),
|
|
array(),
|
|
filemtime( FiveForTheFuture\PATH . '/assets/css/admin.css' )
|
|
);
|
|
|
|
wp_register_script(
|
|
'5ftf-admin',
|
|
plugins_url( 'assets/js/admin.js', __DIR__ ),
|
|
array( 'jquery', 'wp-util' ),
|
|
filemtime( FiveForTheFuture\PATH . '/assets/js/admin.js' ),
|
|
true
|
|
);
|
|
|
|
$pledge_id = is_admin() ? get_the_ID() : absint( $_REQUEST['pledge_id'] ?? 0 );
|
|
$auth_token = sanitize_text_field( $_REQUEST['auth_token'] ?? '' );
|
|
$script_data = array(
|
|
// The global ajaxurl is not set on the frontend.
|
|
'ajaxurl' => admin_url( 'admin-ajax.php', 'relative' ),
|
|
'pledgeId' => $pledge_id,
|
|
'manageNonce' => wp_create_nonce( 'manage-contributors' ),
|
|
'authToken' => $auth_token,
|
|
'removePrompt' => __( 'Are you sure you want to remove this pledge?', 'wporg-5ftf' ),
|
|
);
|
|
wp_add_inline_script(
|
|
'5ftf-admin',
|
|
sprintf(
|
|
'var FiveForTheFuture = JSON.parse( decodeURIComponent( \'%s\' ) );',
|
|
rawurlencode( wp_json_encode( $script_data ) )
|
|
),
|
|
'before'
|
|
);
|
|
|
|
if ( is_admin() ) {
|
|
$current_page = get_current_screen();
|
|
if ( Pledge\CPT_ID === $current_page->id ) {
|
|
wp_enqueue_style( '5ftf-admin' );
|
|
wp_enqueue_script( '5ftf-admin' );
|
|
}
|
|
} else {
|
|
global $post;
|
|
if ( is_a( $post, 'WP_Post' ) ) {
|
|
$pledge_id = absint( $_REQUEST['pledge_id'] ?? 0 );
|
|
$auth_token = sanitize_text_field( $_REQUEST['auth_token'] ?? '' );
|
|
$can_manage = Auth\can_manage_pledge( $pledge_id, $auth_token );
|
|
if ( ! is_wp_error( $can_manage ) && has_shortcode( $post->post_content, '5ftf_pledge_form_manage' ) ) {
|
|
wp_enqueue_script( '5ftf-admin' );
|
|
}
|
|
}
|
|
}
|
|
}
|