felixarntz/plugin-lib

View on GitHub
src/meta.php

Summary

Maintainability
C
1 day
Test Coverage
<?php
/**
 * Meta abstraction class
 *
 * @package Leaves_And_Love\Plugin_Lib
 * @since 1.0.0
 */

namespace Leaves_And_Love\Plugin_Lib;

use Leaves_And_Love\Plugin_Lib\Traits\Container_Service_Trait;

if ( ! class_exists( 'Leaves_And_Love\Plugin_Lib\Meta' ) ) :

    /**
     * Class for Metadata API
     *
     * The class is a wrapper for the WordPress Metadata API.
     *
     * In a multisite setup this class also supports site options and network options as if they were meta.
     * The only difference between options and meta is that an option cannot have multiple values for one key,
     * therefore the method parameters related to uniqueness of values are ignored when using them with a
     * $meta_type of either 'site' or 'network'.
     *
     * @since 1.0.0
     *
     * @method DB db()
     */
    class Meta extends Service {
        use Container_Service_Trait;

        /**
         * The database service definition.
         *
         * @since 1.0.0
         * @static
         * @var string
         */
        protected static $service_db = 'Leaves_And_Love\Plugin_Lib\DB';

        /**
         * Constructor.
         *
         * @since 1.0.0
         *
         * @param string $prefix   The instance prefix.
         * @param array  $services {
         *     Array of service instances.
         *
         *     @type DB            $db             The database class instance.
         *     @type Error_Handler $error_handler The error handler instance.
         * }
         */
        public function __construct( $prefix, $services ) {
            $this->set_prefix( $prefix );
            $this->set_services( $services );
        }

        /**
         * Adds metadata for the specified object.
         *
         * @since 1.0.0
         *
         * @param string $meta_type  Type of object metadata is for (e.g. row, column or module).
         * @param int    $object_id  ID of the object metadata is for.
         * @param string $meta_key   Metadata key.
         * @param mixed  $meta_value Metadata value. Must be serializable if non-scalar.
         * @param bool   $unique     Optional, default is false.
         *                           Whether the specified metadata key should be unique for the object.
         *                           If true, and the object already has a value for the specified metadata key,
         *                           no change will be made.
         * @return int|bool The meta ID or true on success, false on failure.
         */
        public function add( $meta_type, $object_id, $meta_key, $meta_value, $unique = false ) {
            if ( is_multisite() && in_array( $meta_type, array( 'site', 'network' ), true ) ) {
                $callback = 'network' === $meta_type ? 'add_network_option' : 'add_blog_option';
                $result   = call_user_func( $callback, $object_id, $meta_key, $meta_value );
                if ( $result ) {
                    return $this->db()->insert_id;
                }

                return $result;
            }

            if ( $this->is_prefixed_type( $meta_type ) ) {
                $meta_type = $this->db()->get_prefix() . $meta_type;
            }
            return add_metadata( $meta_type, $object_id, $meta_key, $meta_value, $unique );
        }

        /**
         * Updates metadata for the specified object. If no value already exists for the specified object
         * ID and metadata key, the metadata will be added.
         *
         * @since 1.0.0
         *
         * @param string $meta_type  Type of object metadata is for (e.g. row, column or module).
         * @param int    $object_id  ID of the object metadata is for.
         * @param string $meta_key   Metadata key.
         * @param mixed  $meta_value Metadata value. Must be serializable if non-scalar.
         * @param mixed  $prev_value Optional. If specified, only update existing metadata entries with
         *                           the specified value. Otherwise, update all entries.
         * @return int|bool Meta ID if the key didn't exist, true on successful update, false on failure.
         */
        public function update( $meta_type, $object_id, $meta_key, $meta_value, $prev_value = '' ) {
            if ( is_multisite() && in_array( $meta_type, array( 'site', 'network' ), true ) ) {
                $callback = 'network' === $meta_type ? 'update_network_option' : 'update_blog_option';

                $adding = false;
                if ( false === call_user_func( str_replace( 'update_', 'get_', $callback ), $object_id, $meta_key ) ) {
                    $adding = true;
                }

                $result = call_user_func( $callback, $object_id, $meta_key, $meta_value );
                if ( $result && $adding ) {
                    return $this->db()->insert_id;
                }

                return $result;
            }

            if ( $this->is_prefixed_type( $meta_type ) ) {
                $meta_type = $this->db()->get_prefix() . $meta_type;
            }
            return update_metadata( $meta_type, $object_id, $meta_key, $meta_value, $prev_value );
        }

        /**
         * Deletes metadata for the specified object.
         *
         * @since 1.0.0
         *
         * @param string $meta_type  Type of object metadata is for (e.g. row, column or module).
         * @param int    $object_id  ID of the object metadata is for.
         * @param string $meta_key   Metadata key.
         * @param mixed  $meta_value Optional. Metadata value. Must be serializable if non-scalar. If specified, only delete
         *                           metadata entries with this value. Otherwise, delete all entries with the specified meta_key.
         *                           Pass `null, `false`, or an empty string to skip this check. For backward compatibility,
         *                           it is not possible to pass an empty string to delete those entries with an empty string
         *                           for a value.
         * @param bool   $delete_all Optional, default is false. If true, delete matching metadata entries for all objects,
         *                           ignoring the specified object_id. Otherwise, only delete matching metadata entries for
         *                           the specified object_id.
         * @return bool True on successful delete, false on failure.
         */
        public function delete( $meta_type, $object_id, $meta_key, $meta_value = '', $delete_all = false ) {
            if ( is_multisite() && in_array( $meta_type, array( 'site', 'network' ), true ) ) {
                $callback = 'network' === $meta_type ? 'delete_network_option' : 'delete_blog_option';

                return call_user_func( $callback, $object_id, $meta_key );
            }

            if ( $this->is_prefixed_type( $meta_type ) ) {
                $meta_type = $this->db()->get_prefix() . $meta_type;
            }
            return delete_metadata( $meta_type, $object_id, $meta_key, $meta_value, $delete_all );
        }

        /**
         * Retrieves metadata for the specified object.
         *
         * @since 1.0.0
         *
         * @param string $meta_type Type of object metadata is for (e.g. row, column or module).
         * @param int    $object_id ID of the object metadata is for.
         * @param string $meta_key  Optional. Metadata key. If not specified, retrieve all metadata for
         *                          the specified object.
         * @param bool   $single    Optional, default is false.
         *                          If true, return only the first value of the specified meta_key.
         *                          This parameter has no effect if meta_key is not specified.
         * @return mixed Single metadata value, or array of values.
         */
        public function get( $meta_type, $object_id, $meta_key = '', $single = false ) {
            if ( is_multisite() && in_array( $meta_type, array( 'site', 'network' ), true ) ) {
                // Querying all site or network options is not possible here.
                if ( ! $meta_key ) {
                    return array();
                }

                $callback = 'network' === $meta_type ? 'get_network_option' : 'get_blog_option';

                $result = call_user_func( $callback, $object_id, $meta_key );
                if ( ! $single ) {
                    if ( false !== $result ) {
                        return array( $result );
                    }

                    return array();
                }

                return $result;
            }

            if ( $this->is_prefixed_type( $meta_type ) ) {
                $meta_type = $this->db()->get_prefix() . $meta_type;
            }
            $values = get_metadata( $meta_type, $object_id, $meta_key, false );

            // Return false if a single value does not exist.
            if ( $meta_key && $single ) {
                if ( isset( $values[0] ) ) {
                    return $values[0];
                }
                return false;
            }

            return $values;
        }

        /**
         * Determines if a meta key is set for a given object
         *
         * @since 1.0.0
         *
         * @param string $meta_type Type of object metadata is for (e.g. row, column or module).
         * @param int    $object_id ID of the object metadata is for.
         * @param string $meta_key  Metadata key.
         * @return bool True of the key is set, false if not.
         */
        public function exists( $meta_type, $object_id, $meta_key ) {
            if ( is_multisite() ) {
                if ( 'site' === $meta_type ) {
                    if ( false === get_blog_option( $object_id, $meta_key ) ) {
                        return false;
                    }
                    return true;
                } elseif ( 'network' === $meta_type ) {
                    if ( false === get_network_option( $object_id, $meta_key ) ) {
                        return false;
                    }
                    return true;
                }
            }

            if ( $this->is_prefixed_type( $meta_type ) ) {
                $meta_type = $this->db()->get_prefix() . $meta_type;
            }
            return metadata_exists( $meta_type, $object_id, $meta_key );
        }

        /**
         * Deletes all metadata for the specified object.
         *
         * @since 1.0.0
         *
         * @param string $meta_type  Type of object metadata is for (e.g. row, column or module).
         * @param int    $object_id  ID of the object metadata is for.
         * @return bool True on successful delete, false on failure.
         */
        public function delete_all( $meta_type, $object_id ) {
            if ( ! $this->is_prefixed_type( $meta_type ) ) {
                return false;
            }

            $prefixed_meta_type = $this->db()->get_prefix() . $meta_type;

            $meta_ids = $this->db()->get_col( "SELECT meta_id FROM %{$meta_type}meta% WHERE {$prefixed_meta_type}_id = %d", $object_id );
            foreach ( $meta_ids as $meta_id ) {
                delete_metadata_by_mid( $prefixed_meta_type, $meta_id );
            }

            return true;
        }

        /**
         * Updates the metadata cache for the specified objects.
         *
         * @since 1.0.0
         *
         * @param string $meta_type  Type of object metadata is for (e.g., row, column or module).
         * @param array  $object_ids Array of object IDs to update cache for.
         * @return array|false Metadata cache for the specified objects, or false on failure.
         */
        public function update_cache( $meta_type, $object_ids ) {
            if ( $this->is_prefixed_type( $meta_type ) ) {
                $meta_type = $this->db()->get_prefix() . $meta_type;
            }

            return update_meta_cache( $meta_type, $object_ids );
        }

        /**
         * Checks whether a meta type must be prefixed.
         *
         * @since 1.0.0
         *
         * @param string $meta_type Type of object metadata is for (e.g. row, column or module).
         * @return bool True if the meta type must be prefixed, false otherwise.
         */
        protected function is_prefixed_type( $meta_type ) {
            return $this->db()->table_exists( $meta_type . 'meta' );
        }
    }

endif;