<?php
/**
 * Zend Framework (http://framework.zend.com/)
 *
 * @link      http://github.com/zendframework/zf2 for the canonical source repository
 * @copyright Copyright (c) 2005-2013 Zend Technologies USA Inc. (http://www.zend.com)
 * @license   http://framework.zend.com/license/new-bsd New BSD License
 */

namespace Zend\Captcha;

use 
Traversable;
use 
Zend\Validator\AbstractValidator;

/**
 * Base class for Captcha adapters
 *
 * Provides some utility functionality to build on
 */
abstract class AbstractAdapter extends AbstractValidator implements AdapterInterface
{
    
/**
     * Captcha name
     *
     * Useful to generate/check form fields
     *
     * @var string
     */
    
protected $name;

    
/**
     * Captcha options
     *
     * @var array
     */
    
protected $options = array();

    
/**
     * Options to skip when processing options
     * @var array
     */
    
protected $skipOptions = array(
        
'options',
        
'config',
    );

    
/**
     * Get name
     *
     * @return string
     */
    
public function getName()
    {
        return 
$this->name;
    }

    
/**
     * Set name
     *
     * @param string $name
     * @return AbstractAdapter
     */
    
public function setName($name)
    {
        
$this->name $name;
        return 
$this;
    }

    
/**
     * Set single option for the object
     *
     * @param  string $key
     * @param  string $value
     * @return AbstractAdapter
     */
    
public function setOption($key$value)
    {
        if (
in_array(strtolower($key), $this->skipOptions)) {
            return 
$this;
        }

        
$method 'set' ucfirst ($key);
        if (
method_exists ($this$method)) {
            
// Setter exists; use it
            
$this->$method ($value);
            
$this->options[$key] = $value;
        } elseif (
property_exists($this$key)) {
            
// Assume it's metadata
            
$this->$key $value;
            
$this->options[$key] = $value;
        }
        return 
$this;
    }

    
/**
     * Set object state from options array
     *
     * @param  array|Traversable $options
     * @throws Exception\InvalidArgumentException
     * @return AbstractAdapter
     */
    
public function setOptions($options = array())
    {
        if (!
is_array($options) && !$options instanceof Traversable) {
            throw new 
Exception\InvalidArgumentException(__METHOD__ ' expects an array or Traversable');
        }

        foreach (
$options as $key => $value) {
            
$this->setOption($key$value);
        }
        return 
$this;
    }

    
/**
     * Retrieve options representing object state
     *
     * @return array
     */
    
public function getOptions()
    {
        return 
$this->options;
    }

    
/**
     * Get helper name used to render captcha
     *
     * By default, return empty string, indicating no helper needed.
     *
     * @return string
     */
    
public function getHelperName()
    {
        return 
'';
    }
}
validArgumentException(sprintf(
                    
'processor plugin manager must extend %s\ProcessorPluginManager; received %s',
                    
__NAMESPACE__,
                    
is_object($plugins) ? get_class($plugins) : gettype($plugins)
            ));
        }

        
$this->processorPlugins $plugins;
        return 
$this;
    }

    
/**
     * Get processor instance
     *
     * @param string $name
     * @param array|null $options
     * @return Processor\ProcessorInterface
     */
    
public function processorPlugin($name, array $options null)
    {
        return 
$this->getProcessorPluginManager()->get($name$options);
    }

    
/**
     * Add a processor to a logger
     *
     * @param  string|Processor\ProcessorInterface $processor
     * @param  int $priority
     * @param  array|null $options
     * @return Logger
     * @throws Exception\InvalidArgumentException
     */
    
public function addProcessor($processor$priority 1, array $options null)
    {
        if (
is_string($processor)) {
            
$processor $this->processorPlugin($processor$options);
        } elseif (!
$processor instanceof Processor\ProcessorInterface) {
            throw new 
Exception\InvalidArgumentException(sprintf(
                    
'Processor must implement Zend\Log\ProcessorInterface; received "%s"',
                    
is_object($processor) ? get_class($processor) : gettype($processor)
            ));
        }
        
$this->processors->insert($processor$priority);

        return 
$this;
    }

    
/**
     * Get processors
     *
     * @return SplPriorityQueue
     */
    
public function getProcessors()
    {
        return 
$this->processors;
    }

    
/**
     * Add a message as a log entry
     *
     * @param  int $priority
     * @param  mixed $message
     * @param  array|Traversable $extra
     * @return Logger
     * @throws Exception\InvalidArgumentException if message can't be cast to string
     * @throws Exception\InvalidArgumentException if extra can't be iterated over
     * @throws Exception\RuntimeException if no log writer specified
     */
    
public function log($priority$message$extra = array())
    {
        if (!
is_int($priority) || ($priority<0) || ($priority>=count($this->priorities))) {
            throw new 
Exception\InvalidArgumentException(sprintf(
                
'$priority must be an integer > 0 and < %d; received %s',
                
count($this->priorities),
                
var_export($priority1)
            ));
        }
        if (
is_object($message) && !method_exists($message'__toString')) {
            throw new 
Exception\InvalidArgumentException(
                
'$message must implement magic __toString() method'
            
);
        }

        if (!
is_array($extra) && !$extra instanceof Traversable) {
            throw new 
Exception\InvalidArgumentException(
                
'$extra must be an array or implement Traversable'
            
);
        } elseif (
$extra instanceof Traversable) {
            
$extra ArrayUtils::iteratorToArray($extra);
        }

        if (
$this->writers->count() === 0) {
            throw new 
Exception\RuntimeException('No log writer specified');
        }

        
$timestamp = new DateTime();

        if (
is_array($message)) {
            
$message var_export($messagetrue);
        }

        
$event = array(
            
'timestamp'    => $timestamp,
            
'priority'     => (int) $priority,
            
'priorityName' => $this->priorities[$priority],
            
'message'      => (string) $message,
            
'extra'        => $extra
        
);

        foreach (
$this->processors->toArray() as $processor) {
            
$event $processor->process($event);
        }

        foreach (
$this->writers->toArray() as $writer) {
            
$writer->write($event);
        }

        return 
$this;
    }

    
/**
     * @param string $message
     * @param array|Traversable $extra
     * @return Logger
     */
    
public function emerg($message$extra = array())
    {
        return 
$this->log(self::EMERG$message$extra);
    }

    
/**
     * @param string $message
     * @param array|Traversable $extra
     * @return Logger
     */
    
public function alert($message$extra = array())
    {
        return 
$this->log(self::ALERT$message$extra);
    }

    
/**
     * @param string $message
     * @param array|Traversable $extra
     * @return Logger
     */
    
public function crit($message$extra = array())
    {
        return 
$this->log(self::CRIT$message$extra);
    }

    
/**
     * @param string $message
     * @param array|Traversable $extra
     * @return Logger
     */
    
public function err($message$extra = array())
    {
        return 
$this->log(self::ERR$message$extra);
    }

    
/**
     * @param string $message
     * @param array|Traversable $extra
     * @return Logger
     */
    
public function warn($message$extra = array())
    {
        return 
$this->log(self::WARN$message$extra);
    }

    
/**
     * @param string $message
     * @param array|Traversable $extra
     * @return Logger
     */
    
public function notice($message$extra = array())
    {
        return 
$this->log(self::NOTICE$message$extra);
    }

    
/**
     * @param string $message
     * @param array|Traversable $extra
     * @return Logger
     */
    
public function info($message$extra = array())
    {
        return 
$this->log(self::INFO$message$extra);
    }

    
/**
     * @param string $message
     * @param array|Traversable $extra
     * @return Logger
     */
    
public function debug($message$extra = array())
    {
        return 
$this->log(self::DEBUG$message$extra);
    }

    
/**
     * Register logging system as an error handler to log PHP errors
     *
     * @link http://www.php.net/manual/function.set-error-handler.php
     * @param  Logger $logger
     * @param  bool   $continueNativeHandler
     * @return mixed  Returns result of set_error_handler
     * @throws Exception\InvalidArgumentException if logger is null
     */
    
public static function registerErrorHandler(Logger $logger$continueNativeHandler false)
    {
        
// Only register once per instance
        
if (static::$registeredErrorHandler) {
            return 
false;
        }

        
$errorHandlerMap = static::$errorPriorityMap;

        
$previous set_error_handler(function ($level$message$file$line$context)
            use (
$logger$errorHandlerMap$continueNativeHandler)
        {
            
$iniLevel error_reporting();

            if (
$iniLevel $level) {
                if (isset(
Logger::$errorPriorityMap[$level])) {
                    
$priority $errorHandlerMap[$level];
                } else {
                    
$priority Logger::INFO;
                }
                
$logger->log($priority$message, array(
                    
'errno'   => $level,
                    
'file'    => $file,
                    
'line'    => $line,
                    
'context' => $context,
                ));
            }

            return !
$continueNativeHandler;
        });

        static::
$registeredErrorHandler true;
        return 
$previous;
    }

    
/**
     * Unregister error handler
     *
     */
    
public static function unregisterErrorHandler()
    {
        
restore_error_handler();
        static::
$registeredErrorHandler false;
    }

    
/**
     * Register logging system as an exception handler to log PHP exceptions
     *
     * @link http://www.php.net/manual/en/function.set-exception-handler.php
     * @param Logger $logger
     * @return bool
     * @throws Exception\InvalidArgumentException if logger is null
     */
    
public static function registerExceptionHandler(Logger $logger)
    {
        
// Only register once per instance
        
if (static::$registeredExceptionHandler) {
            return 
false;
        }

        if (
$logger === null) {
            throw new 
Exception\InvalidArgumentException('Invalid Logger specified');
        }

        
$errorPriorityMap = static::$errorPriorityMap;

        
set_exception_handler(function ($exception) use ($logger$errorPriorityMap) {
            
$logMessages = array();

            do {
                
$priority Logger::ERR;
                if (
$exception instanceof ErrorException && isset($errorPriorityMap[$exception->getSeverity()])) {
                    
$priority $errorPriorityMap[$exception->getSeverity()];
                }

                
$extra = array(
                    
'file'  => $exception->getFile(),
                    
'line'  => $exception->getLine(),
                    
'trace' => $exception->getTrace(),
                );
                if (isset(
$exception->xdebug_message)) {
                    
$extra['xdebug'] = $exception->xdebug_message;
                }

                
$logMessages[] = array(
                    
'priority' => $priority,
                    
'message'  => $exception->getMessage(),
                    
'extra'    => $extra,
                );
                
$exception $exception->getPrevious();
            } while (
$exception);

            foreach (
array_reverse($logMessages) as $logMessage) {
                
$logger->log($logMessage['priority'], $logMessage['message'], $logMessage['extra']);
            }
        });

        static::
$registeredExceptionHandler true;
        return 
true;
    }

    
/**
     * Unregister exception handler
     */
    
public static function unregisterExceptionHandler()
    {
        
restore_exception_handler();
        static::
$registeredExceptionHandler false;
    }
}