2017-04-20 02:33:27 +03:00
|
|
|
{-# LANGUAGE GADTs #-}
|
2017-04-21 23:56:19 +03:00
|
|
|
module Semantic
|
2017-04-24 17:46:32 +03:00
|
|
|
( diffBlobPairs
|
|
|
|
, diffBlobPair
|
2017-04-21 23:56:19 +03:00
|
|
|
, parseBlobs
|
2017-04-22 00:10:50 +03:00
|
|
|
, parseBlob
|
2017-04-22 00:20:30 +03:00
|
|
|
, parserForLanguage
|
2017-04-21 23:56:19 +03:00
|
|
|
) where
|
2017-04-20 02:33:27 +03:00
|
|
|
|
2017-04-21 01:13:28 +03:00
|
|
|
import Control.Parallel.Strategies
|
2017-04-22 04:06:56 +03:00
|
|
|
import qualified Control.Concurrent.Async as Async
|
|
|
|
import qualified Data.Text as T
|
2017-04-20 02:33:27 +03:00
|
|
|
import Data.Functor.Both
|
2017-04-25 19:02:41 +03:00
|
|
|
import RWS
|
2017-04-20 02:33:27 +03:00
|
|
|
import Data.Record
|
|
|
|
import Diff
|
|
|
|
import Info
|
|
|
|
import Interpreter
|
2017-04-22 04:06:56 +03:00
|
|
|
import Language
|
|
|
|
import Language.Markdown
|
|
|
|
import Parser
|
|
|
|
import Patch
|
2017-04-20 02:33:27 +03:00
|
|
|
import Prologue
|
|
|
|
import Renderer
|
|
|
|
import Source
|
|
|
|
import Syntax
|
|
|
|
import Term
|
2017-04-22 00:20:30 +03:00
|
|
|
import Text.Parser.TreeSitter.C
|
|
|
|
import Text.Parser.TreeSitter.Go
|
|
|
|
import Text.Parser.TreeSitter.JavaScript
|
|
|
|
import Text.Parser.TreeSitter.Ruby
|
|
|
|
import Text.Parser.TreeSitter.TypeScript
|
2017-04-22 04:06:56 +03:00
|
|
|
import TreeSitter
|
2017-04-22 00:20:30 +03:00
|
|
|
|
2017-04-24 18:12:53 +03:00
|
|
|
-- TODO: Shouldn't need to depend on System.FilePath in here, but this is
|
|
|
|
-- currently the way we do language detection.
|
|
|
|
import System.FilePath
|
2017-04-20 02:33:27 +03:00
|
|
|
|
2017-04-21 23:56:19 +03:00
|
|
|
-- This is the primary interface to the Semantic library which provides two
|
|
|
|
-- major classes of functionality: semantic parsing and diffing of source code
|
|
|
|
-- blobs.
|
|
|
|
--
|
2017-04-22 04:06:56 +03:00
|
|
|
-- Design goals:
|
|
|
|
-- - No knowledge of the filesystem or Git.
|
|
|
|
-- - Built in concurrency where appropriate.
|
2017-04-21 23:56:19 +03:00
|
|
|
-- - Easy to consume this interface from other application (e.g a cmdline or web server app).
|
2017-04-20 02:33:27 +03:00
|
|
|
|
2017-04-22 04:06:56 +03:00
|
|
|
-- | Diff a list of SourceBlob pairs to produce ByteString output using the specified renderer.
|
2017-04-24 17:46:32 +03:00
|
|
|
diffBlobPairs :: (Monoid output, StringConv output ByteString) => DiffRenderer DefaultFields output -> [Both SourceBlob] -> IO ByteString
|
|
|
|
diffBlobPairs renderer blobs = do
|
2017-04-22 01:40:37 +03:00
|
|
|
diffs <- Async.mapConcurrently go blobs
|
2017-04-21 20:25:47 +03:00
|
|
|
let diffs' = diffs >>= \ (blobs, diff) -> (,) blobs <$> toList diff
|
2017-04-24 17:37:41 +03:00
|
|
|
toS <$> renderConcurrently (resolveDiffRenderer renderer) (diffs' `using` parTraversable (parTuple2 r0 rdeepseq))
|
2017-04-20 02:33:27 +03:00
|
|
|
where
|
|
|
|
go blobPair = do
|
2017-04-24 17:46:32 +03:00
|
|
|
diff <- diffBlobPair blobPair
|
2017-04-20 02:33:27 +03:00
|
|
|
pure (blobPair, diff)
|
|
|
|
|
2017-04-21 23:56:19 +03:00
|
|
|
-- | Diff a pair of SourceBlobs.
|
2017-04-24 17:46:32 +03:00
|
|
|
diffBlobPair :: Both SourceBlob -> IO (Maybe (Diff (Syntax Text) (Record DefaultFields)))
|
|
|
|
diffBlobPair blobs = do
|
2017-04-22 01:40:37 +03:00
|
|
|
terms <- Async.mapConcurrently parseBlob blobs
|
2017-04-24 19:17:18 +03:00
|
|
|
pure $ case (runJoin blobs, runJoin terms) of
|
|
|
|
((left, right), (a, b)) | nonExistentBlob left && nonExistentBlob right -> Nothing
|
|
|
|
| nonExistentBlob right -> Just . pure $ Delete a
|
|
|
|
| nonExistentBlob left -> Just . pure $ Insert b
|
|
|
|
| otherwise -> Just $ runDiff terms
|
2017-04-20 02:33:27 +03:00
|
|
|
where
|
2017-04-21 20:25:47 +03:00
|
|
|
runDiff terms = stripDiff (runBothWith diffTerms (fmap decorate (terms `using` parTraversable rdeepseq)))
|
2017-04-20 02:33:27 +03:00
|
|
|
decorate = defaultFeatureVectorDecorator getLabel
|
|
|
|
getLabel :: HasField fields Category => TermF (Syntax Text) (Record fields) a -> (Category, Maybe Text)
|
|
|
|
getLabel (h :< t) = (Info.category h, case t of
|
|
|
|
Leaf s -> Just s
|
|
|
|
_ -> Nothing)
|
|
|
|
|
2017-04-21 23:56:19 +03:00
|
|
|
-- | Parse a list of SourceBlobs and use the specified renderer to produce ByteString output.
|
2017-04-20 04:27:36 +03:00
|
|
|
parseBlobs :: (Monoid output, StringConv output ByteString) => ParseTreeRenderer DefaultFields output -> [SourceBlob] -> IO ByteString
|
|
|
|
parseBlobs renderer blobs = do
|
2017-04-20 21:00:02 +03:00
|
|
|
terms <- traverse go blobs
|
2017-04-24 17:37:41 +03:00
|
|
|
toS <$> renderConcurrently (resolveParseTreeRenderer renderer) (terms `using` parTraversable (parTuple2 r0 rdeepseq))
|
2017-04-20 21:00:02 +03:00
|
|
|
where
|
|
|
|
go blob = do
|
2017-04-21 23:56:19 +03:00
|
|
|
term <- parseBlob blob
|
2017-04-21 01:13:28 +03:00
|
|
|
pure (blob, term)
|
2017-04-20 02:33:27 +03:00
|
|
|
|
|
|
|
-- | Parse a SourceBlob.
|
2017-04-21 23:56:19 +03:00
|
|
|
parseBlob :: SourceBlob -> IO (Term (Syntax Text) (Record DefaultFields))
|
|
|
|
parseBlob blob@SourceBlob{..} = parserForFilePath path blob
|
2017-04-22 00:20:30 +03:00
|
|
|
|
|
|
|
-- | Return a parser for a given langauge or the lineByLineParser parser.
|
|
|
|
parserForLanguage :: Maybe Language -> Parser (Syntax Text) (Record DefaultFields)
|
|
|
|
parserForLanguage Nothing = lineByLineParser
|
|
|
|
parserForLanguage (Just language) = case language of
|
|
|
|
C -> treeSitterParser C tree_sitter_c
|
|
|
|
JavaScript -> treeSitterParser JavaScript tree_sitter_javascript
|
|
|
|
TypeScript -> treeSitterParser TypeScript tree_sitter_typescript
|
|
|
|
Markdown -> cmarkParser
|
|
|
|
Ruby -> treeSitterParser Ruby tree_sitter_ruby
|
|
|
|
Language.Go -> treeSitterParser Language.Go tree_sitter_go
|
|
|
|
|
|
|
|
|
2017-04-24 19:17:18 +03:00
|
|
|
-- Internal
|
2017-04-22 00:20:30 +03:00
|
|
|
|
2017-04-24 17:37:41 +03:00
|
|
|
renderConcurrently :: (Monoid output, StringConv output ByteString) => (a -> b -> output) -> [(a, b)] -> IO output
|
|
|
|
renderConcurrently f diffs = do
|
2017-04-22 04:06:56 +03:00
|
|
|
outputs <- Async.mapConcurrently (pure . uncurry f) diffs
|
|
|
|
pure $ mconcat (outputs `using` parTraversable rseq)
|
|
|
|
|
2017-04-22 00:20:30 +03:00
|
|
|
-- | Return a parser based on the FilePath's extension (including the ".").
|
2017-04-24 18:12:53 +03:00
|
|
|
-- | TODO: Remove this.
|
2017-04-22 00:20:30 +03:00
|
|
|
parserForFilePath :: FilePath -> Parser (Syntax Text) (Record DefaultFields)
|
|
|
|
parserForFilePath = parserForLanguage . languageForType . toS . takeExtension
|
|
|
|
|
|
|
|
-- | A fallback parser that treats a file simply as rows of strings.
|
|
|
|
lineByLineParser :: Parser (Syntax Text) (Record DefaultFields)
|
|
|
|
lineByLineParser SourceBlob{..} = pure . cofree . root $ case foldl' annotateLeaves ([], 0) lines of
|
|
|
|
(leaves, _) -> cofree <$> leaves
|
|
|
|
where
|
|
|
|
lines = actualLines source
|
|
|
|
root children = (sourceRange :. Program :. rangeToSourceSpan source sourceRange :. Nil) :< Indexed children
|
|
|
|
sourceRange = Source.totalRange source
|
|
|
|
leaf charIndex line = (Range charIndex (charIndex + T.length line) :. Program :. rangeToSourceSpan source (Range charIndex (charIndex + T.length line)) :. Nil) :< Leaf line
|
|
|
|
annotateLeaves (accum, charIndex) line =
|
|
|
|
(accum <> [ leaf charIndex (Source.toText line) ] , charIndex + Source.length line)
|