Files
root b6df4dbb92 Snapshot: MLS sync fixes, image refresh, plugin/theme updates
MLS plugin fixes from this session:
- Fix silent insert failures: location column NOT NULL was rejecting wpdb->insert calls,
  causing ~18k new properties since Dec 2025 to be lost. Inserts now build raw SQL
  with ST_PointFromText so the spatial column is populated atomically.
- Auto-refresh expired media URLs in MLS_Media_Handler::fetch_and_cache(), guarded by
  a property-level GET_LOCK so concurrent fetches share one API refresh.
- Normalize WP_Error to null in mls_get_property_image() so callers can rely on the
  documented string|null contract.
- Support comma-separated property_type filters in MLS_Query and MLS_Cluster so the
  homepage "View All Commercial" link (?property_type=Commercial+Sale,Land,Farm)
  actually filters correctly.
- Incremental sync now looks back 10 minutes past the latest modification timestamp
  as a safety margin against missed records.
- Smart sync exits silently (info-level, not warning) when a full sync is in progress.

Operational:
- New cron: weekly full sync Sundays at 3 AM (/usr/local/bin/mls-full-sync).
- New cron: hourly 2GB cap on mls-thumbnails/ and cache/transformed-images/
  (/usr/local/bin/mls-image-cache-cap).
- Logrotate config for wp-content/debug.log (2-day retention, daily rotation,
  delaycompress).

Repo policy:
- CLAUDE.md updated with explicit "commit everything except build artifacts" policy.
- .gitignore: untrack runtime image caches and debug.log rotations.

Other modifications in this snapshot are pre-existing in-flight theme/plugin/db_content_updates work.

Co-Authored-By: Claude Opus 4.7 (1M context) <noreply@anthropic.com>
2026-04-29 15:32:23 +00:00

189 lines
4.3 KiB
JavaScript
Executable File

/* global wpforms_challenge_admin, ajaxurl, WPFormsBuilder */
/**
* WPForms Challenge Admin function.
*
* @since 1.5.0
* @since 1.6.2 Challenge v2
*/
'use strict';
var WPFormsChallenge = window.WPFormsChallenge || {};
WPFormsChallenge.admin = window.WPFormsChallenge.admin || ( function( document, window, $ ) {
/**
* Public functions and properties.
*
* @since 1.5.0
*
* @type {object}
*/
var app = {
l10n: wpforms_challenge_admin,
/**
* Start the engine.
*
* @since 1.5.0
*/
init: function() {
$( app.ready );
},
/**
* Document ready.
*
* @since 1.5.0
*/
ready: function() {
app.events();
},
/**
* Register JS events.
*
* @since 1.5.0
*/
events: function() {
$( '.wpforms-challenge-list-block' )
.on( 'click', '.challenge-skip', app.skipChallenge )
.on( 'click', '.challenge-cancel', app.cancelChallenge )
.on( 'click', '.toggle-list', app.toggleList );
},
/**
* Toggle list icon click.
*
* @since 1.5.0
*
* @param {object} e Event object.
*/
toggleList: function( e ) {
var $icon = $( e.target ),
$listBlock = $( '.wpforms-challenge-list-block' );
if ( ! $listBlock.length || ! $icon.length ) {
return;
}
if ( $listBlock.hasClass( 'closed' ) ) {
wpforms_challenge_admin.option.window_closed = '0';
$listBlock.removeClass( 'closed' );
setTimeout( function() {
$listBlock.removeClass( 'transition-back' );
}, 600 );
} else {
wpforms_challenge_admin.option.window_closed = '1';
$listBlock.addClass( 'closed' );
// Add `transition-back` class when the forward transition is completed.
// It is needed to properly implement transitions order for some elements.
setTimeout( function() {
$listBlock.addClass( 'transition-back' );
}, 600 );
}
},
/**
* Skip the Challenge without starting it.
*
* @since 1.5.0
*/
skipChallenge: function() {
var optionData = {
status : 'skipped',
seconds_spent: 0,
seconds_left : app.l10n.minutes_left * 60,
};
$( '.wpforms-challenge' ).remove();
// In the Form Builder, we must also make the Embed button clickable.
$( '#wpforms-embed' ).removeClass( 'wpforms-disabled' );
app.saveChallengeOption( optionData );
},
/**
* Cancel Challenge after starting it.
*
* @since 1.6.2
*/
cancelChallenge: function() {
var core = WPFormsChallenge.core;
core.timer.pause();
/* eslint-disable camelcase */
var optionData = {
status : 'canceled',
seconds_spent: core.timer.getSecondsSpent(),
seconds_left : core.timer.getSecondsLeft(),
feedback_sent: false,
};
/* eslint-enable */
core.removeChallengeUI();
core.clearLocalStorage();
if ( typeof WPFormsBuilder !== 'undefined' ) {
WPFormsChallenge.admin.saveChallengeOption( optionData )
.done( function() { // Save the form before removing scripts if we're in a WPForms Builder.
if ( localStorage.getItem( 'wpformsChallengeStep' ) !== null ) {
WPFormsBuilder.formSave( false );
}
} ).done( // Remove scripts related to challenge.
$( '#wpforms-challenge-admin-js, #wpforms-challenge-core-js, #wpforms-challenge-admin-js-extra, #wpforms-challenge-builder-js' )
.remove()
);
} else {
WPFormsChallenge.admin.saveChallengeOption( optionData )
.done( app.triggerPageSave ); // Assume we're on form embed page.
}
},
/**
* Set Challenge parameter(s) to Challenge option.
*
* @since 1.5.0
*
* @param {object} optionData Query using option schema keys.
*
* @returns {promise} jQuery.post() promise interface.
*/
saveChallengeOption: function( optionData ) {
var data = {
action : 'wpforms_challenge_save_option',
option_data: optionData,
_wpnonce : app.l10n.nonce,
};
// Save window closed (collapsed) state as well.
data.option_data.window_closed = wpforms_challenge_admin.option.window_closed;
$.extend( wpforms_challenge_admin.option, optionData );
return $.post( ajaxurl, data, function( response ) {
if ( ! response.success ) {
console.error( 'Error saving WPForms Challenge option.' );
}
} );
},
};
// Provide access to public functions/properties.
return app;
}( document, window, jQuery ) );
WPFormsChallenge.admin.init();