Code Monkey home page Code Monkey logo

annotated-cache's Introduction

Repository abandoned 2020-11-27

This repository has been archived since we are not using it anymore internally. Feel free to use it AS-IS, we won't be providing any support anymore.

Build status Insight Installs Packagist

Cache Annotations

Stop worrying about caching, use annotations instead. This PHP library makes it possible to add annotations to your services and handles caching for you. You can use any PSR-6 caching implementation as a caching back-end.

Installation

composer require phpro/annotated-cache

We suggest using php-cache to make it possible to add tags to your cache items.

Bridges

Usage

Example usage:

use Phpro\AnnotatedCache\Factory;

$poolManager = Factory::createPoolManager();
$poolManager->addPool('products', $psr6CachePool);

$cacheHandler = Factory::createCacheHandler($poolManager);
$proxyGenerator = Factory::createProxyGenerator($cacheHandler);

$myService = $proxyGenerator->generate(new My\Service());

We made it as easy as possible to get started with the cache manager. You will need 3 services:

  • PoolManager: contains one or multiple PSR-6 cache pools.
  • CacheHandler: contains the PoolManager and the logic for interacting with the cache pool.
  • ProxyGenerator: wraps your service with an Access Interceptor Value Holder

Example Service

<?php

namespace My;

use Phpro\AnnotatedCache\Annotation\Cacheable;
use Phpro\AnnotatedCache\Annotation\CacheUpdate;
use Phpro\AnnotatedCache\Annotation\CacheEvict;

class Service
{

    /**
     * @Cacheable(pools="products", key="sku", tags="product-detail", ttl=300)
     */
    public function getProduct($sku, $type = 'book')
    {
        // fetch a product from a repository or whatever
        $product = $this->productRepository->getByType($sku, 'book');

        return $product;
    }

    /**
     * @CacheEvict(pools="products", key="product.getSku()", tags="product-overview,product-reports")
     */
    public function removeProduct(Product $product)
    {
        // saving product ...
    }

    /**
     * @CacheUpdate(pools="products", key="product.getSku()", tags="product-detail", ttl=300)
     */
    public function updateProduct(Product $product)
    {
        // saving product....
        return $product;
    }
}

Annotations

The bundle provides the following annotations:

@Cacheable annotation

@Cacheable annotation is used to automatically store the result of a method into the cache.

When a method demarcated with the @Cacheable annotation is called, the bundle checks if an entry exists in the cache before executing the method. If it finds one, the cache result is returned without having to actually execute the method.

If no cache entry is found, the method is executed and the bundle automatically stores its result into the cache.

<?php

namespace My\Manager;

use My\Model\Product;

use Phpro\AnnotatedCache\Annotation\Cacheable;

class ProductManager
{
    /**
     * @Cacheable(pools="products", key="sku", tags="book-detail", ttl=500)
     */
    public function getProduct($sku, $type = 'book')
    {
        // fetch a product from a repository or whatever
        $product = $this->productRepository->getByType($sku, 'book');

        return $product;
    }
}

@CacheEvict annotation

@CacheEvict annotation allows methods to trigger cache population or cache eviction.

When a method is demarcated with @CacheEvict annotation, the bundle will execute the method and then will automatically try to delete the cache entry with the provided key and the provided tags.

<?php

namespace My\Manager;

use My\Model\Product;

use Phpro\AnnotatedCache\Annotation\CacheEvict;

class ProductManager
{
    /**
     * @CacheEvict(pools="products", key="product.getSku()", tags="book-list")
     */
    public function removeProduct(Product $product)
    {
        // saving product ...
    }
}

@CacheUpdate annotation

@CacheUpdate annotation is useful for cases where the cache needs to be updated without interfering with the method execution.

When a method is demarcated with @CacheUpdate annotation, the bundle will always execute the method and then will automatically try to update the cache entry with the method result.

<?php

namespace My\Manager;

use My\Model\Product;

use Phpro\AnnotatedCache\Annotation\CacheUpdate;

class ProductManager
{
    /**
     * @CacheUpdate(pools="products", key="product.getSku()", tags="product-detail", ttl=300)
     */
    public function updateProduct(Product $product)
    {
        // saving product....

        return $product;
    }
}

Expression Language

For key generation, Symfony Expression Language can be used.

/**
 * @CacheUpdate(pools="products", key="product.getSku()")
 */
 public function updateProduct(Product $product)
 {
    // do something
 }

The Expression Language allow you to retrieve any arguments passed to your method and use it to generate the cache key. Note that you also have access to the InterceptionInterface. This means you can add other data like the instance and method to the key. For CacheUpdate and CacheEvict, you will also have access to the returnValue. Here is a little example:

/**
 * @Cacheable(pools="products", key="interception.getMethod() ~ id")
 */
 public function getProduct(int $id)
 {
    // do something
 }

Tags

It is possible to add one or multiple tags to a caching entry. Since this is not a default feature in PSR-6, you will have to implement the cache/taggable-cache TaggablePoolInterface on your cache pool.

Handling Results

It is possible to add a ResultCollectorInterface to the CacheHandler. This way you can provide your application with feedback about what happened with the annotations. By default, a dummy collector will be used that doesn't collect anything.

use Phpro\AnnotatedCache\Factory;
use Phpro\AnnotatedCache\Collector\MemoryResultCollector;

// Instantiate pool

$resultCollector = new MemoryResultCollector();
$cacheHandler = Factory::createCacheHandler($poolManager, $resultCollector);

// Instantiate proxy service ...

$myService->fetchSomethingCached();
$results = $resultCollector->getResults();

The results will be an instance of the ResultCollection which will contain a series of ResultInterface objects. Possible results:

  • EmptyResult (These once are filtered out by the collector)
  • HitResult
  • MissResult
  • EvictResult
  • UpdateResult

Writing your own annotations

It is pretty easy to write your own annotations and let the CacheHandler intercept your own annotation. The only thing you will need to do is write your own implementation of the InterceptorInterface and a custom annotation. Next you can register this custom interceptor on the cache handler and do your own thing.

$cacheHandler = Factory::createCacheHandler($poolManager);
$cacheHandler->addInterceptor($myCustomInterceptor)

About

Submitting bugs and feature requests

Bugs and feature request are tracked on GitHub. Please take a look at our rules before contributing your code.

License

Annotated-cache is licensed under the MIT License - see the LICENSE file for details

Credits

This package is based on the TbbcCacheBundle. It uses exactly the same annotations. The big difference is that this package can be used in any PHP application.

Big ups to the Doctrine, proxy-manager and php-cache project for providing the tools that this package needs!

annotated-cache's People

Contributors

veewee avatar

Stargazers

 avatar  avatar  avatar  avatar  avatar  avatar  avatar  avatar  avatar  avatar  avatar  avatar  avatar  avatar  avatar  avatar  avatar

Watchers

 avatar  avatar  avatar  avatar  avatar  avatar

annotated-cache's Issues

Cache-Namespaces?

Hi there,

Nice looking library. I want to give it try. But I'm new to all the PSR-6 cache handling. So maybe I just didnt understand something about that yet...

Lets say I have a service with 2 methods, each method is a getSomething($key), so I annotate them like this:

/**
 * @Cacheable(pools="products")
 */
public function hasProduct($productCode)
{
...
}

/**
 * @Cacheable(pools="products")
 */
public function getProductName($productCode)
{
....
}

I want to have all arguments as the key, so I omit the key="" attribute. Why is it that even if I use different pools for both methods that I end up having only one key written in my cache backend (which is redis). I thought that the method's name is taken into account when generating the key, or at least the pool's name...

I'm stuck :(

Your code looks awesome and its definitely a better implementation than the one from https://github.com/TheBigBrainsCompany/TbbcCacheBundle (which is also good)

Thanks for any hint
Markus

Add new SymfonyCacheableInterceptor that interacts with Symfony pool manager

Phpro\AnnotatedCache\Interceptor\CacheableInterceptor::interceptSuffix returns Cache\Adapter\Common\CacheItem;

but Symfony\Component\Cache\Adapter interacts only with Symfony\Component\Cache\CacheItem otherwise will skip save to cache

@see Symfony\Component\Cache\Adapter\AbstractAdapter::save()

    public function save(CacheItemInterface $item)
    {
        if (!$item instanceof CacheItem) {
            return false;
        }
        $this->deferred[$item->getKey()] = $item;

        return $this->commit();
    }

Or there should be fix on Symfony side in this method?

Edit: Symfony is logging some debug info via static method CacheItem::log, thats missing in standardized Cache\Adapter\Common\CacheItem;

Edit2: I have created new SymfonyInterceptor
https://github.com/insekticid/phpro-annotated-cache-symfony-cache

Recommend Projects

  • React photo React

    A declarative, efficient, and flexible JavaScript library for building user interfaces.

  • Vue.js photo Vue.js

    ๐Ÿ–– Vue.js is a progressive, incrementally-adoptable JavaScript framework for building UI on the web.

  • Typescript photo Typescript

    TypeScript is a superset of JavaScript that compiles to clean JavaScript output.

  • TensorFlow photo TensorFlow

    An Open Source Machine Learning Framework for Everyone

  • Django photo Django

    The Web framework for perfectionists with deadlines.

  • D3 photo D3

    Bring data to life with SVG, Canvas and HTML. ๐Ÿ“Š๐Ÿ“ˆ๐ŸŽ‰

Recommend Topics

  • javascript

    JavaScript (JS) is a lightweight interpreted programming language with first-class functions.

  • web

    Some thing interesting about web. New door for the world.

  • server

    A server is a program made to process requests and deliver data to clients.

  • Machine learning

    Machine learning is a way of modeling and interpreting data that allows a piece of software to respond intelligently.

  • Game

    Some thing interesting about game, make everyone happy.

Recommend Org

  • Facebook photo Facebook

    We are working to build community through open source technology. NB: members must have two-factor auth.

  • Microsoft photo Microsoft

    Open source projects and samples from Microsoft.

  • Google photo Google

    Google โค๏ธ Open Source for everyone.

  • D3 photo D3

    Data-Driven Documents codes.