-- -*- coding: utf-8; mode: haskell; -*- -- File: library/Language/Ninja/IR/Build.hs -- -- License: -- Copyright 2017 Awake Security -- -- Licensed under the Apache License, Version 2.0 (the "License"); -- you may not use this file except in compliance with the License. -- You may obtain a copy of the License at -- -- http://www.apache.org/licenses/LICENSE-2.0 -- -- Unless required by applicable law or agreed to in writing, software -- distributed under the License is distributed on an "AS IS" BASIS, -- WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. -- See the License for the specific language governing permissions and -- limitations under the License. {-# LANGUAGE ConstraintKinds #-} {-# LANGUAGE DeriveGeneric #-} {-# LANGUAGE FlexibleContexts #-} {-# LANGUAGE FlexibleInstances #-} {-# LANGUAGE KindSignatures #-} {-# LANGUAGE MultiParamTypeClasses #-} {-# LANGUAGE OverloadedStrings #-} {-# LANGUAGE RecordWildCards #-} {-# LANGUAGE UndecidableInstances #-} -- | -- Module : Language.Ninja.IR.Build -- Copyright : Copyright 2017 Awake Security -- License : Apache-2.0 -- Maintainer : opensource@awakesecurity.com -- Stability : experimental -- -- A datatype for Ninja @build@ declarations. -- -- @since 0.1.0 module Language.Ninja.IR.Build ( -- * @Build@ Build, makeBuild, buildRule, buildOuts, buildDeps , BuildConstraint ) where import qualified Control.Lens as Lens import Data.Aeson ((.:), (.=)) import qualified Data.Aeson as Aeson import Data.Text (Text) import Data.HashSet (HashSet) import qualified Data.HashSet as HS import Control.DeepSeq (NFData) import Data.Hashable (Hashable) import GHC.Generics (Generic) import qualified Test.SmallCheck.Series as SC import GHC.Exts (Constraint) import Language.Ninja.IR.Rule (Rule) import Language.Ninja.IR.Target (Dependency, Output) import Flow ((|>)) -------------------------------------------------------------------------------- -- | A Ninja @build@ declaration, as documented -- . -- -- @since 0.1.0 data Build = MkBuild { _buildRule :: !Rule , _buildOuts :: !(HashSet Output) , _buildDeps :: !(HashSet Dependency) } deriving (Eq, Show, Generic) -- | Construct a default 'Build' from the given 'Rule' -- -- @since 0.1.0 {-# INLINE makeBuild #-} makeBuild :: Rule -> Build makeBuild rule = MkBuild { _buildRule = rule , _buildOuts = HS.empty , _buildDeps = HS.empty } -- | The rule to execute when building any of the outputs. -- -- @since 0.1.0 {-# INLINE buildRule #-} buildRule :: Lens.Lens' Build Rule buildRule = Lens.lens _buildRule $ \(MkBuild {..}) x -> MkBuild { _buildRule = x, .. } -- | The outputs that are built as a result of rule execution. -- -- @since 0.1.0 {-# INLINE buildOuts #-} buildOuts :: Lens.Lens' Build (HashSet Output) buildOuts = Lens.lens _buildOuts $ \(MkBuild {..}) x -> MkBuild { _buildOuts = x, .. } -- | The dependencies that must be satisfied before this can be built. -- -- @since 0.1.0 {-# INLINE buildDeps #-} buildDeps :: Lens.Lens' Build (HashSet Dependency) buildDeps = Lens.lens _buildDeps $ \(MkBuild {..}) x -> MkBuild { _buildDeps = x, .. } -- | Converts to @{rule: …, outputs: …, dependencies: …}@. -- -- @since 0.1.0 instance Aeson.ToJSON Build where toJSON (MkBuild {..}) = [ "rule" .= _buildRule , "outputs" .= _buildOuts , "dependencies" .= _buildDeps ] |> Aeson.object -- | Inverse of the 'Aeson.ToJSON' instance. -- -- @since 0.1.0 instance Aeson.FromJSON Build where parseJSON = (Aeson.withObject "Build" $ \o -> do _buildRule <- (o .: "rule") >>= pure _buildOuts <- (o .: "outputs") >>= pure _buildDeps <- (o .: "dependencies") >>= pure pure (MkBuild {..})) -- | Default 'Hashable' instance via 'Generic'. -- -- @since 0.1.0 instance Hashable Build -- | Default 'NFData' instance via 'Generic'. -- -- @since 0.1.0 instance NFData Build -- | Default 'SC.Serial' instance via 'Generic'. -- -- @since 0.1.0 instance (Monad m, BuildConstraint (SC.Serial m)) => SC.Serial m Build -- | Default 'SC.CoSerial' instance via 'Generic'. -- -- @since 0.1.0 instance (Monad m, BuildConstraint (SC.CoSerial m)) => SC.CoSerial m Build -- | The set of constraints required for a given constraint to be automatically -- computed for a 'Build'. -- -- @since 0.1.0 type BuildConstraint (c :: * -> Constraint) = ( c Text , c (HashSet Output) , c (HashSet Dependency) ) --------------------------------------------------------------------------------