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>
This commit is contained in:
root
2026-04-29 15:32:23 +00:00
parent 57b752f54e
commit b6df4dbb92
5385 changed files with 838580 additions and 2416 deletions
@@ -0,0 +1,133 @@
/* eslint-disable camelcase */
/* global ajaxurl, wpforms_education_pointers_payment */
// noinspection ES6ConvertVarToLetConst
/**
* Module for handling education pointers related to payments in WPForms.
*
* @since 1.8.8
*/
// eslint-disable-next-line no-var
var WPFormsPointersPayment = window.WPFormsPointersPayment || ( function( document, window, $, l10n ) {
/**
* Elements holder.
*
* @since 1.8.8
*
* @type {Object}
*/
const el = {};
/**
* Runtime variables.
*
* @since 1.8.8
*
* @type {Object}
*/
const vars = {
/**
* Unique ID for the pointer.
*
* @since 1.8.8
*/
pointerId: l10n.pointer,
/**
* Cryptographic token for validating authorized Ajax data exchange.
*
* @since 1.8.8
*/
nonce: l10n.nonce,
};
/**
* Public functions and properties.
*
* @since 1.8.8
*/
const app = {
/**
* Start the engine.
*
* @since 1.8.8
*/
init() {
$( app.ready );
},
/**
* Document ready.
*
* @since 1.8.8
*/
ready() {
app.setup();
app.bindEvents();
},
/**
* Setup. Prepare some variables.
*
* @since 1.8.8
*/
setup() {
// Cache DOM elements.
el.$document = $( document );
},
/**
* Bind events.
*
* @since 1.8.8
*/
bindEvents() {
el.$document.on( 'click', '#toplevel_page_wpforms-overview [href$="-payments"], #wpforms-education-pointers-payments', app.handleOnClick );
},
/**
* Callback for clicking on the action link.
*
* @since 1.8.8
*
* @param {Object} event An event which takes place in the DOM.
*/
handleOnClick( event ) {
// Prevent the default action.
event.preventDefault();
const $this = $( this );
// Get the href attribute.
const href = $this.attr( 'href' );
// Return early if href is missing.
if ( ! href ) {
return;
}
// Hide the pointer before redirecting.
$this.closest( '.wp-pointer-content' ).parent().hide();
// Send AJAX request.
$.post(
ajaxurl,
{
pointer_id: vars.pointerId,
_ajax_nonce: vars.nonce,
action: 'wpforms_education_pointers_engagement',
}
).done( function() {
window.location.href = href;
} );
},
};
// Provide access to public functions/properties.
return app;
}( document, window, jQuery, wpforms_education_pointers_payment ) );
// Initialize.
WPFormsPointersPayment.init();
@@ -0,0 +1 @@
var WPFormsPointersPayment=window.WPFormsPointersPayment||((e,t,o,n)=>{let i={},r={pointerId:n.pointer,nonce:n.nonce},a={init(){o(a.ready)},ready(){a.setup(),a.bindEvents()},setup(){i.$document=o(e)},bindEvents(){i.$document.on("click",'#toplevel_page_wpforms-overview [href$="-payments"], #wpforms-education-pointers-payments',a.handleOnClick)},handleOnClick(e){e.preventDefault();e=o(this);let n=e.attr("href");n&&(e.closest(".wp-pointer-content").parent().hide(),o.post(ajaxurl,{pointer_id:r.pointerId,_ajax_nonce:r.nonce,action:"wpforms_education_pointers_engagement"}).done(function(){t.location.href=n}))}};return a})(document,window,jQuery,wpforms_education_pointers_payment);WPFormsPointersPayment.init();