hakyll/src/Hakyll/Core/Resource/Provider.hs

100 lines
3.7 KiB
Haskell
Raw Normal View History

2010-12-24 07:42:05 +00:00
-- | This module provides an API for resource providers. Resource providers
-- allow Hakyll to get content from resources; the type of resource depends on
-- the concrete instance.
--
-- A resource is represented by the 'Resource' type. This is basically just a
-- newtype wrapper around 'Identifier' -- but it has an important effect: it
-- guarantees that a resource with this identifier can be provided by one or
-- more resource providers.
--
-- Therefore, it is not recommended to read files directly -- you should use the
-- provided 'Resource' methods.
--
2011-04-05 20:14:49 +00:00
module Hakyll.Core.Resource.Provider
( ResourceProvider (..)
2011-04-12 09:43:12 +00:00
, makeResourceProvider
2010-12-31 12:28:31 +00:00
, resourceExists
2010-12-26 15:22:05 +00:00
, resourceDigest
, resourceModified
2010-12-24 07:42:05 +00:00
) where
2011-04-12 09:43:12 +00:00
import Control.Applicative ((<$>))
import Control.Concurrent (MVar, readMVar, modifyMVar_, newMVar)
import Data.Map (Map)
import qualified Data.Map as M
2010-12-24 07:42:05 +00:00
2011-08-27 10:16:03 +00:00
import qualified Crypto.Hash.MD5 as MD5
import qualified Data.ByteString as B
2010-12-26 15:12:24 +00:00
import qualified Data.ByteString.Lazy as LB
2010-12-26 15:22:05 +00:00
import Hakyll.Core.Store
2011-04-05 20:14:49 +00:00
import Hakyll.Core.Resource
2010-12-24 07:42:05 +00:00
-- | A value responsible for retrieving and listing resources
--
data ResourceProvider = ResourceProvider
{ -- | A list of all resources this provider is able to provide
2011-05-06 08:28:35 +00:00
resourceList :: [Resource]
2010-12-24 07:42:05 +00:00
, -- | Retrieve a certain resource as string
2011-05-06 08:28:35 +00:00
resourceString :: Resource -> IO String
2010-12-26 15:12:24 +00:00
, -- | Retrieve a certain resource as lazy bytestring
2011-05-06 08:28:35 +00:00
resourceLBS :: Resource -> IO LB.ByteString
, -- | Cache keeping track of modified items
2011-05-06 08:28:35 +00:00
resourceModifiedCache :: MVar (Map Resource Bool)
2010-12-24 07:42:05 +00:00
}
2010-12-26 15:22:05 +00:00
2011-04-12 09:43:12 +00:00
-- | Create a resource provider
--
makeResourceProvider :: [Resource] -- ^ Resource list
-> (Resource -> IO String) -- ^ String reader
-> (Resource -> IO LB.ByteString) -- ^ ByteString reader
-> IO ResourceProvider -- ^ Resulting provider
makeResourceProvider l s b = ResourceProvider l s b <$> newMVar M.empty
-- | Check if a given identifier has a resource
2010-12-31 12:28:31 +00:00
--
resourceExists :: ResourceProvider -> Resource -> Bool
resourceExists provider = flip elem $ resourceList provider
2010-12-31 12:28:31 +00:00
2010-12-26 15:22:05 +00:00
-- | Retrieve a digest for a given resource
--
2011-08-27 10:16:03 +00:00
resourceDigest :: ResourceProvider -> Resource -> IO B.ByteString
resourceDigest provider = fmap MD5.hashlazy . resourceLBS provider
-- | Check if a resource was modified
--
2011-04-06 12:40:36 +00:00
resourceModified :: ResourceProvider -> Store -> Resource -> IO Bool
resourceModified provider store resource = do
cache <- readMVar mvar
case M.lookup resource cache of
-- Already in the cache
Just m -> return m
-- Not yet in the cache, check digests (if it exists)
Nothing -> do
m <- if resourceExists provider resource
2011-04-06 12:40:36 +00:00
then digestModified provider store resource
else return False
modifyMVar_ mvar (return . M.insert resource m)
return m
where
mvar = resourceModifiedCache provider
-- | Check if a resource digest was modified
--
2011-04-06 12:40:36 +00:00
digestModified :: ResourceProvider -> Store -> Resource -> IO Bool
digestModified provider store resource = do
-- Get the latest seen digest from the store
lastDigest <- storeGet store itemName identifier
-- Calculate the digest for the resource
newDigest <- resourceDigest provider resource
-- Check digests
2011-05-17 08:57:37 +00:00
if Found newDigest == lastDigest
-- All is fine, not modified
then return False
-- Resource modified; store new digest
else do storeSet store itemName identifier newDigest
return True
where
identifier = toIdentifier resource
itemName = "Hakyll.Core.ResourceProvider.digestModified"