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,70 @@
<?php
/**
* Render a connection.
*
* @since 1.9.3
*
* @var string $slug Provider slug.
*/
// Exit if accessed directly.
if ( ! defined( 'ABSPATH' ) ) {
exit;
}
?>
<div class="wpforms-builder-provider-connection" data-connection_id="{{ data.connection.id }}">
<input type="hidden" class="wpforms-builder-provider-connection-id"
name="providers[{{ data.provider }}][{{ data.connection.id }}][id]"
value="{{ data.connection.id }}">
<div class="wpforms-builder-provider-connection-title">
{{ data.connection.name }}
<button class="wpforms-builder-provider-connection-delete js-wpforms-builder-provider-connection-delete" type="button">
<i class="fa fa-trash-o"></i>
</button>
<input type="hidden"
id="wpforms-builder-constant-contact-v3-provider-{{ data.connection.id }}-name"
name="providers[{{ data.provider }}][{{ data.connection.id }}][name]"
value="{{ data.connection.name }}">
</div>
<div class="wpforms-builder-provider-connection-block wpforms-builder-constant-contact-v3-provider-accounts">
<h4><?php esc_html_e( 'Select Account', 'wpforms-lite' ); ?><span class="required">*</span></h4>
<select class="js-wpforms-builder-constant-contact-v3-provider-connection-account wpforms-required" name="providers[{{ data.provider }}][{{ data.connection.id }}][account_id]"<# if ( _.isEmpty( data.accounts ) ) { #> disabled<# } #>>
<option value="" selected disabled>--- <?php esc_html_e( 'Select Account', 'wpforms-lite' ); ?> ---</option>
<# _.each( data.accounts, function( account, account_id ) { #>
<option value="{{ account_id }}" data-option_id="{{ account['option_id'] }}"
<# if ( account_id === data.connection.account_id ) { #> selected<# } #>>
{{ account.label }}
</option>
<# } ); #>
</select>
</div>
<div class="wpforms-builder-provider-connection-block wpforms-builder-constant-contact-v3-provider-actions">
<h4><?php esc_html_e( 'Action To Perform', 'wpforms-lite' ); ?><span class="required">*</span></h4>
<select class="js-wpforms-builder-constant-contact-v3-provider-connection-action wpforms-required"
id="wpforms-builder-constant-contact-v3-provider-{{ data.connection.id }}-action"
<# if ( _.isEmpty( data.connection.account_id ) ) { #>disabled<# } #>
name="providers[<?php echo esc_attr( $slug ); ?>][{{ data.connection.id }}][action]">
<option value=""<# if ( _.isEmpty( data.connection.action ) ) { #> selected<# } #>>
<?php esc_html_e( '--- Select Action ---', 'wpforms-lite' ); ?>
</option>
<# _.each( data.actions, function( label, name ) { #>
<option value="{{ name }}"<# if ( name === data.connection.action ) { #> selected<# } #>>
{{ label }}
</option>
<# } ); #>
</select>
</div>
<!-- Here is where sub-templates will put its compiled HTML. -->
<div class="wpforms-builder-constant-contact-v3-provider-actions-data" style="margin-bottom: 20px;"></div>
{{{ data.conditional }}}
</div>
@@ -0,0 +1,12 @@
<?php
// Exit if accessed directly.
if ( ! defined( 'ABSPATH' ) ) {
exit;
}
?>
<div class="wpforms-builder-provider-connections-error wpforms-hidden">
<span class="wpforms-builder-provider-connections-error-message">
<?php esc_html_e( 'Something went wrong while performing an AJAX request.', 'wpforms-lite' ); ?>
</span>
</div>
@@ -0,0 +1,27 @@
<?php
// Exit if accessed directly.
if ( ! defined( 'ABSPATH' ) ) {
exit;
}
?>
<div class="wpforms-builder-provider-connection-block">
<h4>{{ data.field.label }}<# if ( data.field.required ) { #><span class="required">*</span><# } #></h4>
<select
class="wpforms-builder-constant-contact-v3-provider-connection-{{data.name}} <# if ( data.field.map ) { #> wpforms-field-map-select<# } #><# if ( data.field.required ) { #> wpforms-required<# } #>"
name="providers[{{ data.provider.slug }}][{{ data.connection.id }}][{{ data.name }}]"
<# if ( data.field.map ) { #>
data-field-map-allowed="{{ data.field.map }}"
data-field-map-placeholder="<?php esc_html_e( '--- Select Form Field ---', 'wpforms-lite' ); ?>"
<# } #>
>
<# fieldValue = data.connection[data.name] ?? ''; #>
<option value="">{{ data.field.placeholder }}</option>
<# _.each( data.options, function( option, key ) {
selected = fieldValue.toString() === option.id.toString(); #>
<option value="{{ option.id }}" <# if ( selected ) { #> selected<# } #> >
{{ option.label }}
</option>
<# } ) #>
</select>
</div>