<?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\Server;

use 
Zend\Stdlib\ErrorHandler;

/**
 * \Zend\Server\Cache: cache server definitions
 */
class Cache
{
    
/**
     * @var array Methods to skip when caching server
     */
    
protected static $skipMethods = array();

    
/**
     * Cache a file containing the dispatch list.
     *
     * Serializes the server definition stores the information
     * in $filename.
     *
     * Returns false on any error (typically, inability to write to file), true
     * on success.
     *
     * @param  string $filename
     * @param  \Zend\Server\Server $server
     * @return bool
     */
    
public static function save($filenameServer $server)
    {
        if (!
is_string($filename)
            || (!
file_exists($filename) && !is_writable(dirname($filename))))
        {
            return 
false;
        }

        
$methods $server->getFunctions();

        if (
$methods instanceof Definition) {
            
$definition = new Definition();
            foreach (
$methods as $method) {
                if (
in_array($method->getName(), static::$skipMethods)) {
                    continue;
                }
                
$definition->addMethod($method);
            }
            
$methods $definition;
        }

        
ErrorHandler::start();
        
$test file_put_contents($filenameserialize($methods));
        
ErrorHandler::stop();
        if (
=== $test) {
            return 
false;
        }

        return 
true;
    }

    
/**
     * Load server definition from a file
     *
     * Unserializes a stored server definition from $filename. Returns false if
     * it fails in any way, true on success.
     *
     * Useful to prevent needing to build the server definition on each
     * request. Sample usage:
     *
     * <code>
     * if (!Zend\Server\Cache::get($filename, $server)) {
     *     require_once 'Some/Service/ServiceClass.php';
     *     require_once 'Another/Service/ServiceClass.php';
     *
     *     // Attach Some\Service\ServiceClass with namespace 'some'
     *     $server->attach('Some\Service\ServiceClass', 'some');
     *
     *     // Attach Another\Service\ServiceClass with namespace 'another'
     *     $server->attach('Another\Service\ServiceClass', 'another');
     *
     *     Zend\Server\Cache::save($filename, $server);
     * }
     *
     * $response = $server->handle();
     * echo $response;
     * </code>
     *
     * @param  string $filename
     * @param  \Zend\Server\Server $server
     * @return bool
     */
    
public static function get($filenameServer $server)
    {
        if (!
is_string($filename)
            || !
file_exists($filename)
            || !
is_readable($filename))
        {
            return 
false;
        }

        
ErrorHandler::start();
        
$dispatch file_get_contents($filename);
        
ErrorHandler::stop();
        if (
false === $dispatch) {
            return 
false;
        }

        
ErrorHandler::start(E_NOTICE);
        
$dispatchArray unserialize($dispatch);
        
ErrorHandler::stop();
        if (
false === $dispatchArray) {
            return 
false;
        }

        
$server->loadFunctions($dispatchArray);

        return 
true;
    }

    
/**
     * Remove a cache file
     *
     * @param  string $filename
     * @return bool
     */
    
public static function delete($filename)
    {
        if (
is_string($filename) && file_exists($filename)) {
            
unlink($filename);
            return 
true;
        }

        return 
false;
    }
}