PluginProbe ʕ •ᴥ•ʔ
CommerceBird – AI Command Center, ERP Integrations & B2B for WooCommerce (Zoho, Exact Online). / 2.5.2
CommerceBird – AI Command Center, ERP Integrations & B2B for WooCommerce (Zoho, Exact Online). v2.5.2
3.0.3 3.0.2 3.0.1 trunk 2.2.14 2.2.15 2.2.16 2.2.17 2.2.18 2.2.19 2.3.0 2.3.1 2.3.10 2.3.11 2.3.12 2.3.13 2.3.14 2.3.2 2.3.3 2.3.4 2.3.5 2.3.6 2.3.7 2.3.8 2.3.9 2.4.0 2.4.1 2.4.2 2.4.3 2.4.4 2.4.5 2.4.6 2.5.0 2.5.1 2.5.2 2.6.0 2.6.1 2.6.2 2.6.3 2.6.4 2.6.5 2.7.0 2.7.1 2.7.2 2.7.3 2.7.4 2.7.5 2.7.6 2.7.7 2.7.8 2.7.9 2.7.91 2.7.92 2.7.93 2.8.0 2.8.1 2.8.2 2.8.3 2.8.4 2.8.5 2.9.0 2.9.1 2.9.2 2.9.3 3.0.0
commercebird / commercebird.php
commercebird Last commit date
admin 8 months ago includes 8 months ago languages 1 year ago vendor 8 months ago LICENSE 1 year ago commercebird.php 8 months ago composer.json 9 months ago data-sync.php 9 months ago index.php 1 year ago readme.txt 8 months ago wp-dependencies.json 8 months ago
commercebird.php
427 lines
1 <?php
2 /**
3 * Plugin Name: CommerceBird
4 * Plugin URI: https://commercebird.com
5 * Author: CommerceBird
6 * Description: This plugin helps you get the most of CommerceBird by allowing you to upload product images, use integrations like Zoho Inventory, Zoho CRM, Exact Online and more. Requires a subscription at CommerceBird.com.
7 * Version: 2.5.2
8 * Requires PHP: 8.2
9 * Requires Plugins: woocommerce
10 * Requires at least: 6.5
11 * Tested up to: 6.8.3
12 * Text Domain: commercebird
13 * Domain Path: /languages
14 *
15 * License: GNU General Public License v3.0
16 * License URI: http://www.gnu.org/licenses/gpl-3.0.html
17 *
18 * @category Fulfillment
19 * @package CommerceBird
20 * @author Fawad Tiemoerie <info@commercebird.com>
21 * @copyright Copyright (c) 2024, CommerceBird
22 * @license https://www.gnu.org/licenses/gpl-3.0.html GPL-3.0-or-later
23 *
24 * WC requires at least: 9.4.0
25 * WC tested up to: 10.2.2
26 */
27
28 if ( ! defined( 'ABSPATH' ) ) {
29 exit; // Exit if accessed directly.
30 }
31
32 if ( ! defined( 'CMBIRD_VERSION' ) ) {
33 define( 'CMBIRD_VERSION', '2.5.2' );
34 }
35 if ( ! defined( 'CMBIRD_PATH' ) ) {
36 define( 'CMBIRD_PATH', plugin_dir_path( __FILE__ ) );
37 }
38 if ( ! defined( 'CMBIRD_URL' ) ) {
39 define( 'CMBIRD_URL', plugin_dir_url( __FILE__ ) );
40 }
41 if ( ! defined( 'CMBIRD_MENU_SLUG' ) ) {
42 define( 'CMBIRD_MENU_SLUG', 'commercebird-app' );
43 }
44
45 require_once CMBIRD_PATH . 'includes/woo-functions.php';
46 require_once CMBIRD_PATH . 'includes/sync/order-backend.php';
47 require_once CMBIRD_PATH . 'data-sync.php';
48
49 require __DIR__ . '/vendor/autoload.php';
50
51 use Automattic\WooCommerce\Utilities\FeaturesUtil;
52 use CommerceBird\Admin\Actions\Sync\ExactOnlineSync;
53 use CommerceBird\Admin\Actions\Sync\ZohoCRMSync;
54 use CommerceBird\API\CMBird_APIs;
55 use CommerceBird\API\CreateOrderWebhook;
56 use CommerceBird\API\Exact;
57 use CommerceBird\API\ProductWebhook;
58 use CommerceBird\API\ShippingWebhook;
59 use CommerceBird\API\Zoho;
60 use CommerceBird\Plugin;
61
62 /*
63 |--------------------------------------------------------------------------
64 | Activation, deactivation and uninstall event.
65 |--------------------------------------------------------------------------
66 */
67
68 register_activation_hook( __FILE__, array( Plugin::class, 'activate' ) );
69 register_deactivation_hook( __FILE__, array( Plugin::class, 'deactivate' ) );
70 register_uninstall_hook( __FILE__, array( Plugin::class, 'uninstall' ) );
71
72 /*
73 |--------------------------------------------------------------------------
74 | Load the plugin translations
75 |--------------------------------------------------------------------------
76 */
77 add_action( 'init', 'cmbird_load_textdomain' );
78 function cmbird_load_textdomain() {
79 load_plugin_textdomain( 'commercebird', false, dirname( plugin_basename( __FILE__ ) ) . '/languages/' );
80 }
81
82 /** Loading Purchase Order Class
83 *
84 * @since 1.0.0
85 */
86 function cmbird_purchase_order_class() {
87 if ( class_exists( 'WooCommerce' ) ) {
88 new CMBIRD_Purchase_Order();
89 }
90 }
91 add_action( 'woocommerce_init', 'cmbird_purchase_order_class' );
92 add_action( 'init', array( CMBIRD_PO_Admin_Manager::class, 'init' ), 11 );
93
94 /*
95 |--------------------------------------------------------------------------
96 | Start the plugin
97 |--------------------------------------------------------------------------
98 */
99 Plugin::init();
100
101 /**
102 * Declaring compatibility for WooCommerce HPOS
103 */
104 add_action(
105 'before_woocommerce_init',
106 function () {
107 if ( class_exists( FeaturesUtil::class ) ) {
108 FeaturesUtil::declare_compatibility( 'custom_order_tables', __FILE__, true );
109 }
110 }
111 );
112
113 /*
114 --------------------------------------------------------------------------
115 | Install required dependencies
116 |--------------------------------------------------------------------------
117 */
118 add_action(
119 'plugins_loaded',
120 static function () {
121 // Only if Premium plan is active via transient subscription_details.
122 $subscription = get_transient( 'subscription_details' );
123 if ( 'active' === $subscription['status'] ) {
124 $premium_plans = array( 'Premium', 'PremiumPlus' );
125 if ( in_array( $subscription['plan'], $premium_plans, true ) ) {
126 if ( class_exists( 'ACF' ) ) {
127 return;
128 }
129 WP_Dependency_Installer::instance( __DIR__ )->run();
130 }
131 }
132 }
133 );
134
135 /**
136 * Hooks for WC Action Scheduler to import or export products
137 */
138 $import_products = new CMBIRD_Products_ZI();
139 $import_pricelist = new CMBIRD_Pricelist_ZI();
140 $product_class = new CMBIRD_Products_ZI_Export();
141 $common_class = new CMBIRD_Common_Functions();
142 $contact_class = new CMBIRD_Contact_ZI();
143 $category_class = new CMBIRD_Categories_ZI();
144 $import_pricelist->wc_b2b_groups();
145 add_action( 'import_group_items_cron', array( $import_products, 'sync_groupitem_recursively' ), 10, 2 );
146 add_action( 'import_simple_items_cron', array( $import_products, 'sync_item_recursively' ), 10, 2 );
147 add_action( 'import_variable_product_cron', array( $import_products, 'import_variable_product_variations' ), 10, 2 );
148 add_action( 'sync_zi_product_cron', array( $product_class, 'cmbird_zi_products_prepare_sync' ), 10, 2 );
149 add_action( 'sync_zi_pricelist', array( $import_pricelist, 'zi_get_pricelist' ), 10, 2 );
150 add_action( 'sync_zi_order', array( $common_class, 'cmbird_orders_prepare_sync' ), 10, 2 );
151 add_action( 'sync_zi_import_contacts', array( $contact_class, 'cmbird_get_zoho_contacts' ), 10, 2 );
152 add_action( 'cmbird_zi_category_cron', array( $category_class, 'cmbird_zi_category_sync_call' ), 10 );
153 // add action to set the zoho rate limit option exceeded to false.
154 add_action( 'cmbird_common', array( CMBIRD_Common_Functions::class, 'set_zoho_rate_limit_option' ) );
155 // Zoho CRM Hooks.
156 add_action( 'sync_zcrm_order', array( $common_class, 'cmbird_orders_prepare_sync' ) );
157 add_action( 'sync_zcrm_contact', array( ZohoCRMSync::class, 'cmbird_zcrm_contact_sync' ) );
158 // Exact Online Hooks.
159 add_action( 'cmbird_sync_eo', array( ExactOnlineSync::class, 'sync' ), 10, 3 );
160 add_action( 'cmbird_exact_online_sync_orders', array( ExactOnlineSync::class, 'sync_orders_via_cron' ) );
161 add_action( 'cmbird_payment_status', array( ExactOnlineSync::class, 'cmbird_payment_status' ), 10, 1 );
162 add_action( 'cmbird_eo_get_payment_statuses', array( ExactOnlineSync::class, 'get_payment_status_via_cron' ) );
163 // Callback functions for scheduled action to process product or customer chunk.
164 add_action(
165 'cmbird_process_product_chunk',
166 function ( $args ) {
167 if ( ! is_array( $args ) || empty( $args['transient_key'] ) ) {
168 return;
169 }
170 $transient_key = $args['transient_key'];
171 $import_products = $args['import_products'] ?? false;
172 $chunked_products = get_transient( $transient_key );
173 if ( $chunked_products ) {
174 $sync = new ExactOnlineSync();
175 $sync->sync( 'product', $chunked_products, (bool) $import_products );
176 // Remove transient after processing.
177 delete_transient( $transient_key );
178 }
179 },
180 10,
181 1
182 );
183 add_action(
184 'cmbird_process_customer_chunk',
185 function ( $args ) {
186 if ( ! is_array( $args ) || empty( $args['transient_key'] ) ) {
187 return;
188 }
189 $transient_key = $args['transient_key'];
190 $import_customers = $args['import_customers'] ?? false;
191 $chunked_customers = get_transient( $transient_key );
192 if ( $chunked_customers ) {
193 $sync = new ExactOnlineSync();
194 $sync->sync( 'customer', $chunked_customers, (bool) $import_customers );
195 // Remove transient after processing.
196 delete_transient( $transient_key );
197 }
198 },
199 10,
200 1
201 );
202 // Zoho CRM Hooks.
203 add_action( 'init', array( ZohoCRMSync::class, 'refresh_token' ) );
204
205 // add classes to REST API.
206 add_action(
207 'rest_api_init',
208 function () {
209 new Zoho();
210 new Exact();
211 new ProductWebhook();
212 new ShippingWebhook();
213 new CreateOrderWebhook();
214 new CMBird_APIs();
215 $po_controller = new CMBIRD_REST_Shop_Purchase_Controller();
216 $po_controller->register_routes();
217 }
218 );
219
220 add_action(
221 'save_post',
222 function ( $post_id, $post ) {
223 if ( 'wcb2b_group' === $post->post_type ) {
224 delete_transient( 'wc_b2b_groups' );
225 }
226 },
227 10,
228 2
229 );
230
231 /**
232 * Perform actions when the plugin is updated
233 *
234 * @param string $upgrader_object
235 * @param array $options
236 * @return void
237 */
238 add_action( 'upgrader_process_complete', 'cmbird_update_plugin_tasks', 10, 2 );
239
240 /**
241 * Perform tasks when the plugin is updated.
242 *
243 * @param \WP_Upgrader $upgrader_object - Upgrader object.
244 * @param array $options - Options array.
245 *
246 * @see https://developer.wordpress.org/reference/hooks/upgrader_process_complete/
247 */
248 function cmbird_update_plugin_tasks( $upgrader_object, $options ) {
249 $this_plugin = plugin_basename( __FILE__ );
250
251 if ( 'update' === $options['action'] && 'plugin' === $options['type'] ) {
252 foreach ( $options['plugins'] as $plugin ) {
253 if ( $plugin === $this_plugin ) {
254 // Perform tasks when the plugin is updated.
255 if ( defined( 'WP_CLI' ) && WP_CLI ) {
256 WP_CLI::add_command(
257 'cmbird:clean-zoho-images',
258 function () {
259 $result = cmbird_delete_zoho_orphaned_attachments();
260 WP_CLI::log( wp_json_encode( $result, JSON_PRETTY_PRINT ) );
261 }
262 );
263 }
264 }
265 }
266 }
267 }
268
269 /**
270 * Delete orphaned image attachments whose files no longer exist on disk.
271 * Checks all files in the uploads directory and its subfolders.
272 *
273 * @return array { deleted_posts, deleted_postmeta, deleted_terms, scanned, kept_exists }
274 */
275 function cmbird_delete_zoho_orphaned_attachments(): array {
276 global $wpdb;
277
278 // Only allow in admin/CLI for safety (optional but recommended).
279 if ( ! ( is_admin() || ( defined( 'WP_CLI' ) && WP_CLI ) ) ) {
280 return array(
281 'error' => 'Not allowed outside admin/CLI.',
282 );
283 }
284
285 // 1) Collect all attachment IDs (not just zoho_image).
286 $ids = $wpdb->get_col(
287 $wpdb->prepare(
288 "SELECT p.ID
289 FROM {$wpdb->posts} p
290 INNER JOIN {$wpdb->postmeta} pm
291 ON pm.post_id = p.ID AND pm.meta_key = '_wp_attached_file'
292 WHERE p.post_type = 'attachment'
293 AND p.post_mime_type LIKE %s",
294 'image/%'
295 )
296 );
297
298 if ( empty( $ids ) ) {
299 return array(
300 'deleted_posts' => 0,
301 'deleted_postmeta' => 0,
302 'deleted_terms' => 0,
303 'scanned' => 0,
304 'kept_exists' => 0,
305 'message' => 'No image attachments were found.',
306 );
307 }
308
309 // 2) Safety: keep only IDs where the actual file is missing on disk.
310 $uploads = wp_get_upload_dir();
311 $basedir = trailingslashit( $uploads['basedir'] );
312 $missing = array();
313 $kept_exist = 0;
314
315 // Fetch the relative paths for the candidate IDs.
316 $id_count = count( $ids );
317 if ( 0 === $id_count ) {
318 return array(
319 'deleted_posts' => 0,
320 'deleted_postmeta' => 0,
321 'deleted_terms' => 0,
322 'scanned' => 0,
323 'kept_exists' => 0,
324 'message' => 'No image attachments were found.',
325 );
326 }
327
328 $placeholders = implode( ',', array_fill( 0, $id_count, '%d' ) );
329 $query = "SELECT pm.post_id AS id, pm.meta_value AS rel_path
330 FROM {$wpdb->postmeta} pm
331 WHERE pm.meta_key = '_wp_attached_file'
332 AND pm.post_id IN ($placeholders)";
333
334 $paths = $wpdb->get_results(
335 $wpdb->prepare( $query, ...$ids ), // phpcs:ignore WordPress.DB.PreparedSQL.NotPrepared.
336 ARRAY_A
337 );
338
339 // Map paths for quick lookup.
340 $map = array();
341 foreach ( $paths as $row ) {
342 $map[ (int) $row['id'] ] = $row['rel_path'];
343 }
344
345 foreach ( $ids as $id ) {
346 $rel = $map[ $id ] ?? '';
347 $abs = $rel ? $basedir . ltrim( $rel, '/\\' ) : '';
348 if ( empty( $rel ) || ! file_exists( $abs ) ) {
349 $missing[] = (int) $id;
350 } else {
351 ++$kept_exist;
352 }
353 }
354
355 if ( empty( $missing ) ) {
356 return array(
357 'deleted_posts' => 0,
358 'deleted_postmeta' => 0,
359 'deleted_terms' => 0,
360 'scanned' => count( $ids ),
361 'kept_exists' => $kept_exist,
362 'message' => 'All referenced files still exist; nothing to delete.',
363 );
364 }
365
366 // Deletion helpers.
367 $delete_in = function ( string $table, string $col, array $id_list ) use ( $wpdb ): int {
368 $id_list = array_map( 'intval', $id_list );
369 if ( empty( $id_list ) ) {
370 return 0;
371 }
372 // Sanitize table and column names.
373 $table = preg_replace( '/[^a-zA-Z0-9_]/', '', $table );
374 $col = preg_replace( '/[^a-zA-Z0-9_]/', '', $col );
375 $affected = 0;
376 $chunks = array_chunk( $id_list, 500 );
377 foreach ( $chunks as $chunk ) {
378 $placeholders = implode( ',', array_fill( 0, count( $chunk ), '%d' ) );
379 $query = "DELETE FROM `$table` WHERE `$col` IN ($placeholders)";
380 $res = $wpdb->query( $wpdb->prepare( $query, ...$chunk ) ); // phpcs:ignore WordPress.DB.PreparedSQL.NotPrepared.
381 $affected += (int) $res;
382 }
383 return $affected;
384 };
385
386 $deleted_posts = 0;
387 $deleted_meta = 0;
388 $deleted_terms = 0;
389 $total_to_delete = count( $missing );
390 $in_transaction = false;
391
392 // 3) Transaction (optional, best effort; will no-op if MySQL engine doesn't support).
393 $wpdb->query( 'START TRANSACTION' );
394 $in_transaction = true;
395
396 try {
397 // Delete post meta.
398 $deleted_meta = $delete_in( $wpdb->postmeta, 'post_id', $missing );
399 // Delete term relationships (usually none for attachments, but safe).
400 $deleted_terms = $delete_in( $wpdb->term_relationships, 'object_id', $missing );
401 // Delete the posts.
402 $deleted_posts = $delete_in( $wpdb->posts, 'ID', $missing );
403
404 $wpdb->query( 'COMMIT' );
405 $in_transaction = false;
406
407 return array(
408 'deleted_posts' => $deleted_posts,
409 'deleted_postmeta' => $deleted_meta,
410 'deleted_terms' => $deleted_terms,
411 'scanned' => count( $ids ),
412 'kept_exists' => $kept_exist,
413 'deleted_requested' => $total_to_delete,
414 'message' => 'Deleted orphaned image attachments and related data.',
415 );
416
417 } catch ( \Throwable $e ) {
418 if ( $in_transaction ) {
419 $wpdb->query( 'ROLLBACK' );
420 }
421 return array(
422 'error' => 'Deletion failed: ' . $e->getMessage(),
423 'scanned' => count( $ids ),
424 );
425 }
426 }
427