rules_core.eval.inc

Contains rules core integration needed during evaluation.

File

modules/rules_core.eval.inc

View source
<?php


/**
 * @file
 * Contains rules core integration needed during evaluation.
 *
 * @addtogroup rules
 *
 * @{
 */

/**
 * Action and condition callback: Invokes a rules component.
 *
 * We do not use the execute() method, but handle executing ourself. That way
 * we can utilize the existing state for saving passed variables.
 */
function rules_element_invoke_component($arguments, RulesPlugin $element) {
    $info = $element->info();
    $state = $arguments['state'];
    $wrapped_args = $state->currentArguments;
    if ($component = rules_get_cache('comp_' . $info['#config_name'])) {
        $replacements = array(
            '%label' => $component->label(),
            '@plugin' => $component->plugin(),
        );
        // Handle recursion prevention.
        if ($state->isBlocked($component)) {
            return rules_log('Not evaluating @plugin %label to prevent recursion.', $replacements, RulesLog::INFO, $component);
        }
        $state->block($component);
        rules_log('Evaluating @plugin %label.', $replacements, RulesLog::INFO, $component, TRUE);
        module_invoke_all('rules_config_execute', $component);
        // Manually create a new evaluation state and evaluate the component.
        $args = array_intersect_key($wrapped_args, $component->parameterInfo());
        $new_state = $component->setUpState($wrapped_args);
        $return = $component->evaluate($new_state);
        // Care for the right return value in case we have to provide vars.
        if ($component instanceof RulesActionInterface && !empty($info['provides'])) {
            $return = array();
            foreach ($info['provides'] as $var => $var_info) {
                $return[$var] = $new_state->get($var);
            }
        }
        // Now merge the info about to be saved variables in the parent state.
        $state->mergeSaveVariables($new_state, $component, $element->settings);
        $state->unblock($component);
        // Cleanup the state, what saves not mergeable variables now.
        $new_state->cleanup();
        rules_log('Finished evaluation of @plugin %label.', $replacements, RulesLog::INFO, $component, FALSE);
        return $return;
    }
    else {
        throw new RulesEvaluationException('Unable to get the component %name', array(
            '%name' => $info['#config_name'],
        ), $element, RulesLog::ERROR);
    }
}

/**
 * A class implementing a rules input evaluator processing date input.
 *
 * This is needed to treat relative date inputs for strtotime() correctly.
 * Consider for example "now".
 */
class RulesDateInputEvaluator extends RulesDataInputEvaluator {
    const DATE_REGEX_LOOSE = '/^(\\d{4})-?(\\d{2})-?(\\d{2})([T\\s]?(\\d{2}):?(\\d{2}):?(\\d{2})?)?$/';
    
    /**
     * Overrides RulesDataInputEvaluator::prepare().
     */
    public function prepare($text, $var_info) {
        if (is_numeric($text)) {
            // Let rules skip this input evaluators in case it's already a timestamp.
            $this->setting = NULL;
        }
    }
    
    /**
     * Overrides RulesDataInputEvaluator::evaluate().
     */
    public function evaluate($text, $options, RulesState $state) {
        return self::gmstrtotime($text);
    }
    
    /**
     * Convert a time string to a GMT (UTC) unix timestamp.
     */
    public static function gmstrtotime($date) {
        // Pass the current timestamp in UTC to ensure the retrieved time is UTC.
        return strtotime($date, time());
    }
    
    /**
     * Determine whether the given date string specifies a fixed date.
     */
    public static function isFixedDateString($date) {
        return is_string($date) && preg_match(self::DATE_REGEX_LOOSE, $date);
    }

}

/**
 * A class implementing a rules input evaluator processing URI inputs.
 *
 * Makes sure URIs are absolute and path aliases get applied.
 */
class RulesURIInputEvaluator extends RulesDataInputEvaluator {
    
    /**
     * Overrides RulesDataInputEvaluator::prepare().
     */
    public function prepare($uri, $var_info) {
        if (!isset($this->processor) && valid_url($uri, TRUE)) {
            // Only process if another evaluator is used or the url is not absolute.
            $this->setting = NULL;
        }
    }
    
    /**
     * Overrides RulesDataInputEvaluator::evaluate().
     */
    public function evaluate($uri, $options, RulesState $state) {
        if (!url_is_external($uri)) {
            // Extract the path and build the URL using the url() function, so URL
            // aliases are applied and query parameters and fragments get handled.
            $url = drupal_parse_url($uri);
            $url_options = array(
                'absolute' => TRUE,
            );
            $url_options['query'] = $url['query'];
            $url_options['fragment'] = $url['fragment'];
            return url($url['path'], $url_options);
        }
        elseif (valid_url($uri)) {
            return $uri;
        }
        throw new RulesEvaluationException('Input evaluation generated an invalid URI.', array(), NULL, RulesLog::WARN);
    }

}

/**
 * A data processor for applying date offsets.
 */
class RulesDateOffsetProcessor extends RulesDataProcessor {
    
    /**
     * Overrides RulesDataProcessor::form().
     */
    protected static function form($settings, $var_info) {
        $settings += array(
            'value' => '',
        );
        $form = array(
            '#type' => 'fieldset',
            '#title' => t('Add offset'),
            '#collapsible' => TRUE,
            '#collapsed' => empty($settings['value']),
            '#description' => t('Add an offset to the selected date.'),
        );
        $form['value'] = array(
            '#type' => 'rules_duration',
            '#title' => t('Offset'),
            '#description' => t('Note that you can also specify negative numbers.'),
            '#default_value' => $settings['value'],
            '#weight' => 5,
        );
        return $form;
    }
    
    /**
     * Overrides RulesDataProcessor::process().
     */
    public function process($value, $info, RulesState $state, RulesPlugin $element) {
        $value = isset($this->processor) ? $this->processor
            ->process($value, $info, $state, $element) : $value;
        return RulesDateOffsetProcessor::applyOffset($value, $this->setting['value']);
    }
    
    /**
     * Intelligently applies the given date offset in seconds.
     *
     * Intelligently apply duration values > 1 day, i.e. convert the duration
     * to its biggest possible unit (months, days) and apply it to the date with
     * the given unit. That's necessary as the number of days in a month
     * differs, as well as the number of hours for a day (on DST changes).
     */
    public static function applyOffset($timestamp, $offset) {
        if (abs($offset) >= 86400) {
            // Get the days out of the seconds.
            $days = intval($offset / 86400);
            $sec = $offset % 86400;
            // Get the months out of the number of days.
            $months = intval($days / 30);
            $days = $days % 30;
            // Apply the offset using the DateTime::modify and convert it back to a
            // timestamp.
            $date = date_create("@{$timestamp}");
            $date->modify("{$months} months {$days} days {$sec} seconds");
            return $date->format('U');
        }
        else {
            return $timestamp + $offset;
        }
    }

}

/**
 * A data processor for applying numerical offsets.
 */
class RulesNumericOffsetProcessor extends RulesDataProcessor {
    
    /**
     * Overrides RulesDataProcessor::form().
     */
    protected static function form($settings, $var_info) {
        $settings += array(
            'value' => '',
        );
        $form = array(
            '#type' => 'fieldset',
            '#title' => t('Add offset'),
            '#collapsible' => TRUE,
            '#collapsed' => empty($settings['value']),
            '#description' => t('Add an offset to the selected number. E.g. an offset of "1" adds 1 to the number before it is passed on as argument.'),
        );
        $form['value'] = array(
            '#type' => 'textfield',
            '#title' => t('Offset'),
            '#description' => t('Note that you can also specify negative numbers.'),
            '#default_value' => $settings['value'],
            '#element_validate' => array(
                'rules_ui_element_integer_validate',
            ),
            '#weight' => 5,
        );
        return $form;
    }
    
    /**
     * Overrides RulesDataProcessor::process().
     */
    public function process($value, $info, RulesState $state, RulesPlugin $element) {
        $value = isset($this->processor) ? $this->processor
            ->process($value, $info, $state, $element) : $value;
        return $value + $this->setting['value'];
    }

}

/**
 * A custom wrapper class for vocabularies.
 *
 * This class is capable of loading vocabularies by machine name.
 */
class RulesTaxonomyVocabularyWrapper extends EntityDrupalWrapper {
    
    /**
     * Overridden to support identifying vocabularies by machine names.
     */
    protected function setEntity($data) {
        if (isset($data) && $data !== FALSE && !is_object($data) && !is_numeric($data)) {
            // The vocabulary name has been passed.
            parent::setEntity(taxonomy_vocabulary_machine_name_load($data));
        }
        else {
            parent::setEntity($data);
        }
    }
    
    /**
     * Overridden to permit machine names as values.
     */
    public function validate($value) {
        if (isset($value) && is_string($value)) {
            return TRUE;
        }
        return parent::validate($value);
    }

}

/**
 * @} End of "addtogroup rules"
 */

Related topics

Functions

Title Deprecated Summary
rules_element_invoke_component Action and condition callback: Invokes a rules component.

Classes

Title Deprecated Summary
RulesDateInputEvaluator A class implementing a rules input evaluator processing date input.
RulesDateOffsetProcessor A data processor for applying date offsets.
RulesNumericOffsetProcessor A data processor for applying numerical offsets.
RulesTaxonomyVocabularyWrapper A custom wrapper class for vocabularies.
RulesURIInputEvaluator A class implementing a rules input evaluator processing URI inputs.