2017-09-18 07:23:14 +03:00
|
|
|
|
{-# LANGUAGE DataKinds, DefaultSignatures, GADTs, RankNTypes, TypeOperators, UndecidableInstances #-}
|
2015-11-18 03:20:52 +03:00
|
|
|
|
module Algorithm where
|
2015-11-18 03:24:01 +03:00
|
|
|
|
|
2017-09-18 17:51:40 +03:00
|
|
|
|
import Control.Applicative (liftA2)
|
2017-08-05 01:49:21 +03:00
|
|
|
|
import Control.Monad (guard, join)
|
2017-09-09 13:23:57 +03:00
|
|
|
|
import Control.Monad.Free.Freer
|
2017-06-13 21:49:22 +03:00
|
|
|
|
import Data.Functor.Classes
|
2017-08-14 23:42:13 +03:00
|
|
|
|
import Data.List.NonEmpty (NonEmpty(..))
|
2017-07-28 21:37:02 +03:00
|
|
|
|
import Data.Maybe
|
2017-08-05 01:49:21 +03:00
|
|
|
|
import Data.Proxy
|
2017-02-23 20:25:45 +03:00
|
|
|
|
import Data.These
|
2017-07-21 00:15:01 +03:00
|
|
|
|
import Data.Union
|
|
|
|
|
import Diff
|
|
|
|
|
import GHC.Generics
|
|
|
|
|
import Term
|
2016-08-04 01:12:31 +03:00
|
|
|
|
|
2017-02-21 18:26:27 +03:00
|
|
|
|
-- | A single step in a diffing algorithm, parameterized by the types of terms, diffs, and the result of the applicable algorithm.
|
2017-09-14 22:17:22 +03:00
|
|
|
|
data AlgorithmF term diff result where
|
2017-02-23 20:24:31 +03:00
|
|
|
|
-- | Diff two terms with the choice of algorithm left to the interpreter’s discretion.
|
2017-09-14 22:17:22 +03:00
|
|
|
|
Diff :: term ann1 -> term ann2 -> AlgorithmF term (diff ann1 ann2) (diff ann1 ann2)
|
2017-02-21 18:38:10 +03:00
|
|
|
|
-- | Diff two terms recursively in O(n) time, resulting in a single diff node.
|
2017-09-14 22:17:22 +03:00
|
|
|
|
Linear :: term ann1 -> term ann2 -> AlgorithmF term (diff ann1 ann2) (diff ann1 ann2)
|
2017-02-21 18:38:10 +03:00
|
|
|
|
-- | Diff two lists of terms by each element’s similarity in O(n³ log n), resulting in a list of diffs.
|
2017-09-14 22:17:22 +03:00
|
|
|
|
RWS :: [term ann1] -> [term ann2] -> AlgorithmF term (diff ann1 ann2) [diff ann1 ann2]
|
2017-02-21 19:40:13 +03:00
|
|
|
|
-- | Delete a term..
|
2017-09-14 22:17:22 +03:00
|
|
|
|
Delete :: term ann1 -> AlgorithmF term (diff ann1 ann2) (diff ann1 ann2)
|
2017-02-21 19:40:13 +03:00
|
|
|
|
-- | Insert a term.
|
2017-09-14 22:17:22 +03:00
|
|
|
|
Insert :: term ann2 -> AlgorithmF term (diff ann1 ann2) (diff ann1 ann2)
|
2017-02-21 19:40:13 +03:00
|
|
|
|
-- | Replace one term with another.
|
2017-09-14 22:17:22 +03:00
|
|
|
|
Replace :: term ann1 -> term ann2 -> AlgorithmF term (diff ann1 ann2) (diff ann1 ann2)
|
2015-11-18 03:24:01 +03:00
|
|
|
|
|
2016-09-25 09:50:49 +03:00
|
|
|
|
-- | The free applicative for 'AlgorithmF'. This enables us to construct diff values using <$> and <*> notation.
|
2017-09-14 22:17:22 +03:00
|
|
|
|
type Algorithm term diff = Freer (AlgorithmF term diff)
|
2017-02-21 18:44:48 +03:00
|
|
|
|
|
2016-08-04 19:08:34 +03:00
|
|
|
|
|
|
|
|
|
-- DSL
|
|
|
|
|
|
2017-02-23 20:24:31 +03:00
|
|
|
|
-- | Diff two terms without specifying the algorithm to be used.
|
2017-09-14 22:17:22 +03:00
|
|
|
|
diff :: term ann1 -> term ann2 -> Algorithm term (diff ann1 ann2) (diff ann1 ann2)
|
2017-09-09 13:23:57 +03:00
|
|
|
|
diff = (liftF .) . Algorithm.Diff
|
2017-02-23 20:24:31 +03:00
|
|
|
|
|
2017-02-23 20:30:26 +03:00
|
|
|
|
-- | Diff a These of terms without specifying the algorithm to be used.
|
2017-09-14 22:17:22 +03:00
|
|
|
|
diffThese :: These (term ann1) (term ann2) -> Algorithm term (diff ann1 ann2) (diff ann1 ann2)
|
2017-02-23 20:25:45 +03:00
|
|
|
|
diffThese = these byDeleting byInserting diff
|
|
|
|
|
|
2017-05-03 23:56:15 +03:00
|
|
|
|
-- | Diff a pair of optional terms without specifying the algorithm to be used.
|
2017-09-14 22:17:22 +03:00
|
|
|
|
diffMaybe :: Maybe (term ann1) -> Maybe (term ann2) -> Algorithm term (diff ann1 ann2) (Maybe (diff ann1 ann2))
|
2017-05-03 23:56:15 +03:00
|
|
|
|
diffMaybe a b = case (a, b) of
|
|
|
|
|
(Just a, Just b) -> Just <$> diff a b
|
|
|
|
|
(Just a, _) -> Just <$> byDeleting a
|
|
|
|
|
(_, Just b) -> Just <$> byInserting b
|
|
|
|
|
_ -> pure Nothing
|
|
|
|
|
|
2017-02-21 19:39:41 +03:00
|
|
|
|
-- | Diff two terms linearly.
|
2017-09-14 22:17:22 +03:00
|
|
|
|
linearly :: term ann1 -> term ann2 -> Algorithm term (diff ann1 ann2) (diff ann1 ann2)
|
2017-02-24 21:28:44 +03:00
|
|
|
|
linearly a b = liftF (Linear a b)
|
2016-08-04 01:38:35 +03:00
|
|
|
|
|
2017-02-21 19:39:41 +03:00
|
|
|
|
-- | Diff two terms using RWS.
|
2017-09-14 22:17:22 +03:00
|
|
|
|
byRWS :: [term ann1] -> [term ann2] -> Algorithm term (diff ann1 ann2) [diff ann1 ann2]
|
2017-02-24 21:28:44 +03:00
|
|
|
|
byRWS a b = liftF (RWS a b)
|
2017-02-21 19:40:13 +03:00
|
|
|
|
|
|
|
|
|
-- | Delete a term.
|
2017-09-14 22:17:22 +03:00
|
|
|
|
byDeleting :: term ann1 -> Algorithm term (diff ann1 ann2) (diff ann1 ann2)
|
2017-02-24 21:28:44 +03:00
|
|
|
|
byDeleting = liftF . Delete
|
2017-02-21 19:40:13 +03:00
|
|
|
|
|
|
|
|
|
-- | Insert a term.
|
2017-09-14 22:17:22 +03:00
|
|
|
|
byInserting :: term ann2 -> Algorithm term (diff ann1 ann2) (diff ann1 ann2)
|
2017-02-24 21:28:44 +03:00
|
|
|
|
byInserting = liftF . Insert
|
2017-02-21 19:40:13 +03:00
|
|
|
|
|
|
|
|
|
-- | Replace one term with another.
|
2017-09-14 22:17:22 +03:00
|
|
|
|
byReplacing :: term ann1 -> term ann2 -> Algorithm term (diff ann1 ann2) (diff ann1 ann2)
|
2017-02-24 21:28:44 +03:00
|
|
|
|
byReplacing = (liftF .) . Replace
|
2017-06-13 21:49:22 +03:00
|
|
|
|
|
|
|
|
|
|
2017-09-14 22:17:22 +03:00
|
|
|
|
instance (Show1 term, Show ann1, Show ann2) => Show1 (AlgorithmF term (diff ann1 ann2)) where
|
2017-06-13 21:49:22 +03:00
|
|
|
|
liftShowsPrec _ _ d algorithm = case algorithm of
|
2017-09-14 01:59:38 +03:00
|
|
|
|
Algorithm.Diff t1 t2 -> showsBinaryWith showsTerm showsTerm "Diff" d t1 t2
|
|
|
|
|
Linear t1 t2 -> showsBinaryWith showsTerm showsTerm "Linear" d t1 t2
|
|
|
|
|
RWS as bs -> showsBinaryWith (liftShowsPrec showsTerm (liftShowList showsPrec showList)) (liftShowsPrec showsTerm (liftShowList showsPrec showList)) "RWS" d as bs
|
|
|
|
|
Delete t1 -> showsUnaryWith showsTerm "Delete" d t1
|
|
|
|
|
Insert t2 -> showsUnaryWith showsTerm "Insert" d t2
|
|
|
|
|
Replace t1 t2 -> showsBinaryWith showsTerm showsTerm "Replace" d t1 t2
|
2017-09-14 22:08:43 +03:00
|
|
|
|
where showsTerm :: (Show1 term, Show ann) => Int -> term ann -> ShowS
|
2017-09-14 01:59:38 +03:00
|
|
|
|
showsTerm = liftShowsPrec showsPrec showList
|
2017-07-21 00:15:01 +03:00
|
|
|
|
|
|
|
|
|
|
2017-07-21 03:21:20 +03:00
|
|
|
|
-- | Diff two terms based on their generic Diffable instances. If the terms are not diffable
|
|
|
|
|
-- (represented by a Nothing diff returned from algorithmFor) replace one term with another.
|
2017-09-14 22:47:37 +03:00
|
|
|
|
algorithmForTerms :: Diffable syntax
|
2017-09-14 16:41:52 +03:00
|
|
|
|
=> Term syntax ann1
|
|
|
|
|
-> Term syntax ann2
|
2017-09-14 22:17:22 +03:00
|
|
|
|
-> Algorithm (Term syntax) (Diff syntax ann1 ann2) (Diff syntax ann1 ann2)
|
2017-09-13 21:06:26 +03:00
|
|
|
|
algorithmForTerms t1 t2 = fromMaybe (byReplacing t1 t2) (algorithmForComparableTerms t1 t2)
|
|
|
|
|
|
2017-09-14 22:47:16 +03:00
|
|
|
|
algorithmForComparableTerms :: Diffable syntax
|
2017-09-14 16:41:52 +03:00
|
|
|
|
=> Term syntax ann1
|
|
|
|
|
-> Term syntax ann2
|
2017-09-14 22:17:22 +03:00
|
|
|
|
-> Maybe (Algorithm (Term syntax) (Diff syntax ann1 ann2) (Diff syntax ann1 ann2))
|
2017-09-13 21:06:26 +03:00
|
|
|
|
algorithmForComparableTerms (Term (In ann1 f1)) (Term (In ann2 f2)) = fmap (merge (ann1, ann2)) <$> algorithmFor f1 f2
|
2017-07-21 00:15:01 +03:00
|
|
|
|
|
2017-07-21 03:21:20 +03:00
|
|
|
|
|
|
|
|
|
-- | A type class for determining what algorithm to use for diffing two terms.
|
2017-07-21 00:15:01 +03:00
|
|
|
|
class Diffable f where
|
2017-09-15 01:04:54 +03:00
|
|
|
|
algorithmFor :: f (term ann1)
|
|
|
|
|
-> f (term ann2)
|
|
|
|
|
-> Maybe (Algorithm term (diff ann1 ann2) (f (diff ann1 ann2)))
|
|
|
|
|
default
|
|
|
|
|
algorithmFor :: (Generic1 f, GDiffable (Rep1 f))
|
|
|
|
|
=> f (term ann1)
|
|
|
|
|
-> f (term ann2)
|
|
|
|
|
-> Maybe (Algorithm term (diff ann1 ann2) (f (diff ann1 ann2)))
|
2017-09-13 18:48:51 +03:00
|
|
|
|
algorithmFor = genericAlgorithmFor
|
|
|
|
|
|
2017-09-14 22:17:22 +03:00
|
|
|
|
genericAlgorithmFor :: (Generic1 f, GDiffable (Rep1 f)) => f (term ann1) -> f (term ann2) -> Maybe (Algorithm term (diff ann1 ann2) (f (diff ann1 ann2)))
|
2017-09-14 22:09:30 +03:00
|
|
|
|
genericAlgorithmFor a b = fmap to1 <$> galgorithmFor (from1 a) (from1 b)
|
2017-09-13 18:48:51 +03:00
|
|
|
|
|
2017-07-21 00:15:01 +03:00
|
|
|
|
|
2017-07-21 03:25:27 +03:00
|
|
|
|
-- | Diff a Union of Syntax terms. Left is the "rest" of the Syntax terms in the Union,
|
2017-07-21 20:50:53 +03:00
|
|
|
|
-- Right is the "head" of the Union. 'weaken' relaxes the Union to allow the possible
|
|
|
|
|
-- diff terms from the "rest" of the Union, and 'inj' adds the diff terms into the Union.
|
2017-07-21 03:21:20 +03:00
|
|
|
|
-- NB: If Left or Right Syntax terms in our Union don't match, we fail fast by returning Nothing.
|
2017-09-18 07:23:14 +03:00
|
|
|
|
instance Apply Diffable fs => Diffable (Union fs) where
|
|
|
|
|
algorithmFor u1 u2 = join (apply2' (Proxy :: Proxy Diffable) (\ inj f1 f2 -> fmap inj <$> algorithmFor f1 f2) u1 u2)
|
2017-07-21 00:15:01 +03:00
|
|
|
|
|
2017-07-21 20:50:24 +03:00
|
|
|
|
-- | Diff two list parameters using RWS.
|
|
|
|
|
instance Diffable [] where
|
|
|
|
|
algorithmFor a b = Just (byRWS a b)
|
|
|
|
|
|
2017-07-21 03:21:20 +03:00
|
|
|
|
-- | A generic type class for diffing two terms defined by the Generic1 interface.
|
2017-09-14 22:09:30 +03:00
|
|
|
|
class GDiffable f where
|
2017-09-14 22:17:22 +03:00
|
|
|
|
galgorithmFor :: f (term ann1) -> f (term ann2) -> Maybe (Algorithm term (diff ann1 ann2) (f (diff ann1 ann2)))
|
2017-07-21 00:15:01 +03:00
|
|
|
|
|
2017-07-21 20:50:53 +03:00
|
|
|
|
-- | Diff two constructors (M1 is the Generic1 newtype for meta-information (possibly related to type constructors, record selectors, and data types))
|
2017-09-14 22:09:30 +03:00
|
|
|
|
instance GDiffable f => GDiffable (M1 i c f) where
|
|
|
|
|
galgorithmFor (M1 a) (M1 b) = fmap M1 <$> galgorithmFor a b
|
2017-07-21 00:15:01 +03:00
|
|
|
|
|
2017-07-21 20:50:53 +03:00
|
|
|
|
-- | Diff the fields of a product type.
|
|
|
|
|
-- i.e. data Foo a b = Foo a b (the 'Foo a b' is captured by 'a :*: b').
|
2017-09-14 22:09:30 +03:00
|
|
|
|
instance (GDiffable f, GDiffable g) => GDiffable (f :*: g) where
|
|
|
|
|
galgorithmFor (a1 :*: b1) (a2 :*: b2) = liftA2 (:*:) <$> galgorithmFor a1 a2 <*> galgorithmFor b1 b2
|
2017-07-21 00:15:01 +03:00
|
|
|
|
|
2017-07-21 20:50:53 +03:00
|
|
|
|
-- | Diff the constructors of a sum type.
|
|
|
|
|
-- i.e. data Foo a = Foo a | Bar a (the 'Foo a' is captured by L1 and 'Bar a' is R1).
|
2017-09-14 22:09:30 +03:00
|
|
|
|
instance (GDiffable f, GDiffable g) => GDiffable (f :+: g) where
|
|
|
|
|
galgorithmFor (L1 a) (L1 b) = fmap L1 <$> galgorithmFor a b
|
|
|
|
|
galgorithmFor (R1 a) (R1 b) = fmap R1 <$> galgorithmFor a b
|
|
|
|
|
galgorithmFor _ _ = Nothing
|
2017-07-21 00:15:01 +03:00
|
|
|
|
|
2017-07-21 03:21:20 +03:00
|
|
|
|
-- | Diff two parameters (Par1 is the Generic1 newtype representing a type parameter).
|
2017-07-21 20:50:53 +03:00
|
|
|
|
-- i.e. data Foo a = Foo a (the 'a' is captured by Par1).
|
2017-09-14 22:09:30 +03:00
|
|
|
|
instance GDiffable Par1 where
|
|
|
|
|
galgorithmFor (Par1 a) (Par1 b) = Just (Par1 <$> linearly a b)
|
2017-07-21 00:15:01 +03:00
|
|
|
|
|
2017-07-21 03:21:20 +03:00
|
|
|
|
-- | Diff two constant parameters (K1 is the Generic1 newtype representing type parameter constants).
|
2017-07-21 20:50:53 +03:00
|
|
|
|
-- i.e. data Foo = Foo Int (the 'Int' is a constant parameter).
|
2017-09-14 22:09:30 +03:00
|
|
|
|
instance Eq c => GDiffable (K1 i c) where
|
|
|
|
|
galgorithmFor (K1 a) (K1 b) = guard (a == b) *> Just (pure (K1 a))
|
2017-07-21 00:15:01 +03:00
|
|
|
|
|
2017-07-21 03:21:20 +03:00
|
|
|
|
-- | Diff two terms whose constructors contain 0 type parameters.
|
|
|
|
|
-- i.e. data Foo = Foo.
|
2017-09-14 22:09:30 +03:00
|
|
|
|
instance GDiffable U1 where
|
|
|
|
|
galgorithmFor _ _ = Just (pure U1)
|
2017-07-21 00:15:01 +03:00
|
|
|
|
|
2017-08-14 23:39:48 +03:00
|
|
|
|
-- | Diff two lists of parameters.
|
2017-09-14 22:09:30 +03:00
|
|
|
|
instance GDiffable (Rec1 []) where
|
|
|
|
|
galgorithmFor a b = Just (Rec1 <$> byRWS (unRec1 a) (unRec1 b))
|
2017-08-14 23:42:13 +03:00
|
|
|
|
|
|
|
|
|
-- | Diff two non-empty lists of parameters.
|
2017-09-14 22:09:30 +03:00
|
|
|
|
instance GDiffable (Rec1 NonEmpty) where
|
|
|
|
|
galgorithmFor (Rec1 (a:|as)) (Rec1 (b:|bs)) = Just $ do
|
2017-08-14 23:42:13 +03:00
|
|
|
|
d:ds <- byRWS (a:as) (b:bs)
|
|
|
|
|
pure (Rec1 (d :| ds))
|