{-# LANGUAGE BlockArguments #-} {-# LANGUAGE DataKinds #-} {-# LANGUAGE DeriveAnyClass #-} {-# LANGUAGE DuplicateRecordFields #-} {-# LANGUAGE LambdaCase #-} {-# LANGUAGE MultiWayIf #-} {-# LANGUAGE OverloadedRecordDot #-} {-# LANGUAGE OverloadedStrings #-} {-# LANGUAGE QuasiQuotes #-} {-# LANGUAGE RecordWildCards #-} {-# LANGUAGE ViewPatterns #-} {-# LANGUAGE NoFieldSelectors #-} {-# OPTIONS_GHC -fno-warn-name-shadowing #-} module Main where import Control.Arrow (first, second) import Control.Concurrent.ParallelIO.Local (parallel, withPool) import Control.Exception (Exception, throw, throwIO) import Control.Monad (unless, when) import Control.Parallel.Strategies (parList, rdeepseq, withStrategy) import Data.Aeson qualified as J import Data.Attoparsec.Text qualified as A import Data.ByteString.Lazy qualified as LB import Data.Digest.Pure.SHA (sha1, showDigest) import Data.Either (partitionEithers) import Data.Function ((&)) import Data.HashMap.Internal qualified as HM import Data.List import Data.List.NonEmpty qualified as N import Data.Map qualified as M import Data.Maybe (catMaybes, fromMaybe, mapMaybe) import Data.Ord (comparing) import Data.Set qualified as S import Data.String (IsString (fromString)) import Data.Text qualified as T import Data.Text.Encoding qualified as T import Data.Text.IO qualified as T import Data.Text.Normalize qualified as T import Data.Time.Clock (getCurrentTime) import Data.Time.Format.ISO8601 (iso8601ParseM) import Debug.Trace import Document qualified as D import GHC.Conc (getNumProcessors, numCapabilities) import Options.Applicative qualified as O import Parallel (streamsOf) import ProbabilityMap qualified as C import Prompt qualified as P import Settings qualified as S import Store qualified as R import System.Directory import System.Environment (getEnv) import System.FilePath import System.IO.Temp (withSystemTempDirectory) import System.Process.Typed import Tag qualified as G import Text.Printf (printf) import Text.RE.Replace qualified as R import Text.RE.TDFA.Text qualified as R import Text.Read (readMaybe) data Args = Args { cmd :: Cmd } data Cmd = Consume { keep :: Bool, inputs :: [FilePath], prompt :: Bool, force :: Bool, language :: Maybe String, auto :: Bool } | Edit { indexNames :: [FilePath] } | List { filters :: [G.Filter], todo :: Bool, view :: Bool, redo :: Bool, edit :: Bool } | Todo | View { indexNames :: [FilePath] } | TopWords | Modify { indexNames :: [FilePath], tags :: [G.Tag], untags :: [G.Tag] } | Check args :: O.Parser Args args = Args <$> cmd cmd :: O.Parser Cmd cmd = O.hsubparser . mconcat $ [ O.command "consume" . O.info consumeCmd $ O.progDesc "Consume document(s)", O.command "edit" . O.info editCmd $ O.progDesc "Edit document(s)", O.command "list" . O.info listCmd $ O.progDesc "List document(s)", O.command "todo" . O.info todoCmd $ O.progDesc "Interactively process new documents", O.command "view" . O.info viewCmd $ O.progDesc "View document(s)", O.command "topwords" . O.info topWordsCmd $ O.progDesc "View probability map per tag", O.command "modify" . O.info modifyCmd $ O.progDesc "Modify document(s)", O.command "check" . O.info checkCmd $ O.progDesc "Check document tags" ] consumeCmd :: O.Parser Cmd consumeCmd = Consume <$> keepArg <*> inputsArg <*> promptArg <*> forceArg <*> languageArg <*> autoArg editCmd :: O.Parser Cmd editCmd = Edit <$> indexNamesArg listCmd :: O.Parser Cmd listCmd = List <$> filtersArg <*> todoArg <*> viewArg <*> redoArg <*> editArg todoCmd :: O.Parser Cmd todoCmd = pure Todo viewCmd :: O.Parser Cmd viewCmd = View <$> indexNamesArg topWordsCmd :: O.Parser Cmd topWordsCmd = pure TopWords modifyCmd :: O.Parser Cmd modifyCmd = Modify <$> indexNamesArg <*> tagsArg <*> untagsArg checkCmd :: O.Parser Cmd checkCmd = pure Check inputsArg :: O.Parser [FilePath] inputsArg = O.many ( O.strArgument ( O.metavar "FILE" <> O.action "file" ) ) keepArg :: O.Parser Bool keepArg = O.switch ( O.long "keep" <> O.short 'k' <> O.help "Keep input document" ) promptArg :: O.Parser Bool promptArg = O.switch ( O.long "prompt" <> O.short 'p' <> O.help "Prompt after consuming document(s)" ) forceArg :: O.Parser Bool forceArg = O.switch ( O.long "force" <> O.short 'f' <> O.help "Force operation, overriding safety checks" ) autoArg :: O.Parser Bool autoArg = O.switch ( O.long "auto" <> O.short 'a' <> O.help "Automatically tag document(s)" ) filtersArg :: O.Parser [G.Filter] filtersArg = O.many $ O.option (O.eitherReader (A.parseOnly G.filterParser . T.pack)) ( O.long "filter" <> O.short 'f' <> O.help "Filter documents by tag" ) tagsArg :: O.Parser [G.Tag] tagsArg = O.many $ O.option (O.eitherReader (A.parseOnly G.tagParser . T.pack)) ( O.long "tag" <> O.help "Tag to add" ) untagsArg :: O.Parser [G.Tag] untagsArg = O.many $ O.option (O.eitherReader (A.parseOnly G.tagParser . T.pack)) ( O.long "untag" <> O.help "Tag to remove" ) languageArg :: O.Parser (Maybe String) languageArg = O.optional ( O.strOption ( O.long "language" <> O.short 'l' <> O.help "Specify document language" ) ) indexNamesArg :: O.Parser [FilePath] indexNamesArg = O.many $ O.strArgument ( O.metavar "ID" <> O.action "file" ) todoArg :: O.Parser Bool todoArg = O.switch ( O.long "todo" <> O.help "Run command `todo` on listed document(s)" ) redoArg :: O.Parser Bool redoArg = O.switch ( O.long "redo" <> O.help "Redo OCR on listed document(s)" ) editArg :: O.Parser Bool editArg = O.switch ( O.long "edit" <> O.help "Run command `edit` on listed document(s)" ) viewArg :: O.Parser Bool viewArg = O.switch ( O.long "view" <> O.help "Run command `view` on listed document(s)" ) main :: IO () main = do settings <- S.readSettings S.writeSettings "apaperless.yaml" settings cwd <- getCurrentDirectory setCurrentDirectory =<< getEnv "APAPERLESS_STORE_DIR" ensureGit ensureDir "originals" ensureDir "index" O.execParser (O.info (args O.<**> O.helper) O.idm) >>= \case Args {cmd = Consume {keep, inputs, force, prompt, language, auto}} -> do indexNames <- parMapM ( consume1 (fromMaybe settings.defaultLanguage language) force keep ) (map (cwd ) inputs) docs <- mapM (D.readDocument . (<.> "json")) indexNames probabilityCache <- C.readProbabilityCache docs' <- if | auto -> processDocuments settings probabilityCache docs | prompt -> processDocumentsInteractively settings probabilityCache docs | otherwise -> pure docs mapM_ ( \doc -> do printf "%s\n" (takeBaseName doc.iFileName) printTags doc ) docs' Args {cmd = Edit {indexNames}} -> do editDocuments =<< mapM (D.readDocument . (<.> "json")) indexNames Args {cmd = List {filters, redo, todo = False, view = False, edit = False}} -> do doRedoIf filters redo mapM_ ( \doc -> do printf "%s\n" (takeBaseName doc.iFileName) printTags doc ) . filter (G.applyFilters filters . (.index.tags)) =<< D.getDocuments Args {cmd = List {filters, redo, edit = True}} -> do doRedoIf filters redo editDocuments . filter (G.applyFilters filters . (.index.tags)) =<< D.getDocuments Args {cmd = List {filters, redo, todo = True}} -> do doRedoIf filters redo allDocs <- D.getDocuments probabilityCache <- C.readProbabilityCache _ <- processDocumentsInteractively settings probabilityCache . filter (G.applyFilters filters . (.index.tags)) $ allDocs pure () Args {cmd = Todo} -> do allDocs <- D.getDocuments probabilityCache <- C.readProbabilityCache _ <- processDocumentsInteractively settings probabilityCache . filter (G.applyFilters [G.filter G.include "todo" Nothing] . (.index.tags)) $ allDocs pure () Args {cmd = List {filters, redo, view = True}} -> do doRedoIf filters redo viewDocuments . filter (G.applyFilters filters . (.index.tags)) =<< D.getDocuments Args {cmd = View {indexNames}} -> do viewDocuments =<< mapM (D.readDocument . (<.> "json")) indexNames Args {cmd = TopWords} -> do probabilityCache <- C.readProbabilityCache mapM_ ( \(tag, xs) -> do print tag mapM_ (\(word, p) -> printf " %s: %.4f\n" word p) xs ) $ map (second (sortBy (comparing (negate . snd)))) $ map (second (filter ((> 0) . snd))) $ M.toList $ M.foldlWithKey ( \wordProbabilityPerTag (word, tag) p -> M.insertWith (++) tag [(word, p)] wordProbabilityPerTag ) M.empty probabilityCache.probabilityMap Args {cmd = Modify {indexNames, tags, untags}} -> do docs <- mapM (D.readDocument . (<.> "json")) indexNames mapM_ ( \doc -> R.replaceDocument (printf "tag %s" (takeBaseName doc.iFilePath)) doc ) (map (addTags tags . removeTags untags) docs) Args {cmd = Check} -> do allDocs <- D.getDocuments probabilityCache <- C.readProbabilityCache mapM_ ( \(doc, tags, tags') -> do printf "%s\n" (takeBaseName doc.iFileName) printTags doc mapM_ (\tag -> printf ("- %s\n") (show tag)) (tags `S.difference` tags') mapM_ (\tag -> printf ("+ %s\n") (show tag)) (tags' `S.difference` tags) ) ( catMaybes ( concat ( withStrategy (parList rdeepseq) ( streamsOf numCapabilities ( map ( \doc -> let tags = S.filter arbitrarilySelectTag doc.index.tags tags' = S.fromList $ autoApplySuggestedTags $ suggestTags settings probabilityCache doc in if tags == tags' then Nothing else Just (doc, tags, tags') ) allDocs ) ) ) ) ) printTags :: D.Document -> IO () printTags doc = mapM_ ( \tag -> case G.tagValue tag of Nothing -> printf "@%s\n" (G.tagKey tag) Just tagValue -> printf "@%s %s\n" (G.tagKey tag) tagValue ) ( filter arbitrarilySelectTag . S.toList $ doc.index.tags `S.union` doc.index.internalTags ) arbitrarilySelectTag :: G.Tag -> Bool arbitrarilySelectTag tag = -- TODO Handle tags hidden by default -- -- - Add `list-default-hide-tags` to `Settings` -- - Add `--show-tags`, `--hide-tags` to `List` not ("paperless-ngx." `T.isPrefixOf` G.tagKey tag) && (G.tagKey tag /= "content") && (G.tagKey tag /= "importedFrom") && (G.tagKey tag /= "language") doRedoIf :: [G.Filter] -> Bool -> IO () doRedoIf filters redo = when redo do parMapM_ doRedo . filter (G.applyFilters filters . (.index.tags)) =<< D.getDocuments where doRedo doc = do content <- ocr doc.index.language doc.oFilePath let doc' = doc {D.index = doc.index {D.content = content}} R.replaceDocument (printf "redo %s" (takeBaseName doc.iFilePath)) doc' processDocumentsInteractively :: S.Settings -> C.ProbabilityCache -> [D.Document] -> IO [D.Document] processDocumentsInteractively settings probabilityCache docs = do mapM (uncurry processDocumentInteractively) (zip [1 :: Int ..] docs) where numDocs = length docs processDocumentInteractively n doc = do choice <- P.prompt ( P.choice ( printf "%s\n%s\n\n(%d/%d) Process this document?" (takeBaseName doc.iFileName) doc.index.shortText n numDocs ) (("f" :: String) N.:| ["p", "s", "v"]) & P.help ( \a -> case a of "f" -> "view full text" "p" -> "process document" "s" -> "skip document" "v" -> "view document" ) ) case choice of "f" -> do printf "%s\n" (takeBaseName doc.iFileName) printf "%s\n" doc.index.content processDocumentInteractively n doc "p" -> tagDocumentInteractively settings probabilityCache doc "s" -> pure doc "v" -> do viewDocuments [doc] processDocumentInteractively n doc viewDocuments :: [D.Document] -> IO () viewDocuments docs = sh_ ( "zathura " <> ( intercalate " " (map (\doc -> printf "'%s'" doc.oFilePath) docs) ) ) editDocuments :: [D.Document] -> IO () editDocuments docs = withSystemTempDirectory "apaperless" $ \tmp -> do let fp doc = tmp takeBaseName doc.iFileName <.> "txt" mapM_ ( \doc -> T.writeFile (fp doc) doc.index.content ) docs sh_ ( "vim " <> ( intercalate " " (map (\doc -> printf "'%s'" (fp doc)) docs) ) ) mapM_ ( \doc -> do content <- T.readFile (fp doc) let doc' = doc {D.index = doc.index {D.content = content}} R.replaceDocument (printf "edit %s" (takeBaseName doc.iFilePath)) doc' pure content ) docs suggestTags :: S.Settings -> C.ProbabilityCache -> D.Document -> [(G.Tag, [G.Tag])] suggestTags settings probabilityCache doc = do flip map settings.suggestedTags $ \suggestedTag -> case suggestedTag of S.SuggestTagByRE tagName searchReplaces -> let tagValues = nub . map snd . sortBy (comparing fst) . concat $ map ( \searchReplace -> mapMaybe ( -- XXX Whys is this so complicated? \match -> do (topCapture, captures) <- R.matchCaptures match pure ( topCapture.captureOffset, foldl ( \template (captureName, captureOrdinal) -> T.replace ("${" <> R.getCaptureName captureName <> "}") (R.capturedText (captures !! (fromEnum captureOrdinal - 1))) template ) (R.getTemplate searchReplace) (HM.toList $ R.captureNames match) ) ) ( R.allMatches ( doc.index.content R.*=~ R.getSearch searchReplace ) ) ) searchReplaces in (G.tag tagName (Just ""), map (G.tag tagName . Just) tagValues) S.SuggestTagByTags tagName -> let tagValues = -- TODO Cache `probabilityCache` -- -- @topic probability-map probabilityCache.probabilityMap & M.filterWithKey ( \(word, tag) _ -> G.tagKey tag == tagName && D.hasWord word doc ) & M.foldlWithKey' ( \scorePerTagValue (_, tag) p -> M.insertWith (+) (G.tagValue tag) p scorePerTagValue ) M.empty & M.toList & sortBy (comparing (negate . snd)) & map fst in (G.tag tagName (Just ""), map (G.tag tagName) tagValues) autoApplySuggestedTags :: [(G.Tag, [G.Tag])] -> [G.Tag] autoApplySuggestedTags = mapMaybe ( \(_, suggestedTags) -> if null suggestedTags then Nothing else Just (head suggestedTags) ) processDocuments :: S.Settings -> C.ProbabilityCache -> [D.Document] -> IO [D.Document] processDocuments settings probabilityCache docs = mapM processDocument docs where processDocument doc = do let tags = S.fromList . autoApplySuggestedTags $ suggestTags settings probabilityCache doc let doc' = doc { D.index = doc.index { D.tags = doc.index.tags `S.union` tags, D.todo = False } } R.replaceDocument (printf "process %s (auto)" doc.iFilePath) doc' pure doc' applyTags :: [Either T.Text G.Tag] -> D.Document -> D.Document applyTags tags' doc = do addTags tags (removeTags untags doc) where (untags, tags) = first (map (\tagKey -> G.tag tagKey Nothing)) $ partitionEithers tags' data AddTagException = AddTagException G.Tag deriving (Show) instance Exception AddTagException addTags :: [G.Tag] -> D.Document -> D.Document addTags = flip (foldl (flip addTag)) addTag :: G.Tag -> D.Document -> D.Document addTag tag doc = if | G.tagKey tag == "todo" -> doc {D.index = doc.index {D.todo = True}} | G.tagKey tag == "addedAt" -> maybe (throw (AddTagException tag)) ( \addedAt -> doc {D.index = doc.index {D.addedAt = addedAt}} ) (iso8601ParseM . T.unpack =<< (G.tagValue tag)) | G.tagKey tag == "modifiedAt" -> maybe (throw (AddTagException tag)) ( \modifiedAt -> doc {D.index = doc.index {D.modifiedAt = Just modifiedAt}} ) (iso8601ParseM . T.unpack =<< (G.tagValue tag)) | G.tagKey tag == "content" -> maybe (throw (AddTagException tag)) ( \content -> doc {D.index = doc.index {D.content = content}} ) (G.tagValue tag) | G.tagKey tag == "language" -> throw (AddTagException tag) | otherwise -> doc { D.index = doc.index { D.tags = G.replace tag doc.index.tags } } data RemoveTagException = RemoveTagException G.Tag deriving (Show) instance Exception RemoveTagException removeTags :: [G.Tag] -> D.Document -> D.Document removeTags = flip (foldl (flip removeTag)) removeTag :: G.Tag -> D.Document -> D.Document removeTag tag doc = if | G.tagKey tag == "todo" -> doc {D.index = doc.index {D.todo = False}} | G.tagKey tag == "addedAt" -> throw (RemoveTagException tag) | G.tagKey tag == "modifiedAt" -> throw (RemoveTagException tag) | G.tagKey tag == "language" -> throw (RemoveTagException tag) | G.tagKey tag == "content" -> throw (RemoveTagException tag) | otherwise -> doc { D.index = doc.index { D.tags = maybe (G.deleteAll (G.tagKey tag)) (\_ -> G.delete tag) (G.tagValue tag) doc.index.tags } } tagDocumentInteractively :: S.Settings -> C.ProbabilityCache -> D.Document -> IO D.Document tagDocumentInteractively settings probabilityCache doc = do let suggestedTags = suggestTags settings probabilityCache doc tags <- mapM (uncurry tagDocumentInteractively') suggestedTags let doc' = (applyTags tags doc) & \doc' -> doc' { D.index = doc'.index {D.todo = False} } R.replaceDocument (printf "process %s (interactive)" doc.iFilePath) doc' pure doc' where tagDocumentInteractively' :: G.Tag -> [G.Tag] -> IO (Either T.Text G.Tag) tagDocumentInteractively' tag tags | Nothing <- G.tagValue tag = do choice <- P.prompt $ P.choice (printf "tag with %s?" (G.tagKey tag)) (("n" :: String) N.:| ["y"]) pure $ if (choice == "y") then Right tag else Left (G.tagKey tag) | Just _ <- G.tagValue tag = do tagValue <- fmap T.pack . P.prompt $ P.string (printf "tag with %s?" (G.tagKey tag)) (mapMaybe (fmap T.unpack . G.tagValue) tags ++ ["-"]) pure $ if tagValue == "-" then Left (G.tagKey tag) else Right (G.tag (G.tagKey tag) (Just tagValue)) ensureGit :: IO () ensureGit = do doesExist <- doesDirectoryExist ".git" unless doesExist $ sh_ ">/dev/null git init --initial-branch main" ensureDir :: FilePath -> IO () ensureDir dirName = createDirectoryIfMissing False dirName debug :: Show a => String -> a -> a debug s x = trace (printf "%s: %s\n" s (show x)) x fileKey :: FilePath -> IO FilePath fileKey filePath = showDigest . sha1 <$> LB.readFile filePath consume1 :: String -> Bool -> Bool -> FilePath -> IO FilePath consume1 language force keep filePath = do fKey <- fileKey filePath let oFilePath = "originals" fKey <.> takeExtension filePath originalExists <- doesFileExist oFilePath when (originalExists && not force) do error (printf "error: error adding %s: duplicate of %s\n" filePath oFilePath) let iFilePath = "index" fKey <.> "json" content <- do content' <- T.normalize T.NFC . T.decodeUtf8 . LB.toStrict <$> sh (printf "pdftotext -layout '%s' -" filePath) let hasText = (not . T.null) . T.strip $ content' if not hasText then ocr language filePath else pure content' addedAt <- getCurrentTime let doc = D.Document (fKey <.> "json") $ D.Index { tags = S.empty, todo = True, modifiedAt = Nothing, .. } R.commitDocument (printf "add %s" (takeFileName filePath)) filePath doc when (not keep) do removeFile filePath pure (takeBaseName iFilePath) data DecodeException = DecodeException FilePath String deriving (Show) instance Exception DecodeException decodeFile :: J.FromJSON a => FilePath -> IO a decodeFile fp = either (throwIO . DecodeException fp) pure . J.eitherDecode =<< LB.readFile fp ocr :: String -> FilePath -> IO T.Text ocr language input = withSystemTempDirectory (takeBaseName input) $ \tmp -> do let fn suffix = tmp takeBaseName input <> suffix pdfInfo <- parsePdfInfo <$> sh (printf "pdfinfo '%s'" input) pdfImages <- parsePdfImages <$> sh (printf "pdfimages -list '%s'" input) let isScan = length pdfImages == pdfInfo.numPages && all ((pdfInfo.pageSize ==) . imageSize) pdfImages if isScan then sh_ (printf "pdfimages '%s' '%s' -tiff" input (fn "")) else sh_ (printf "pdftoppm '%s' '%s' -r 300 -tiff" input (fn "-%d.pdf")) imageFiles <- sort <$> listDirectory tmp -- XXX add DPI information to image so that resulting pdf preserves DPI parMapM_ ( \(pdfImage, imageFile) -> sh_ ( printf "convert -density %dx%d -units PixelsPerInch '%s' '%s'" pdfImage.xPpi pdfImage.yPpi (tmp imageFile) (tmp imageFile) ) ) (zip pdfImages imageFiles) T.unlines <$> mapM (ocr1 language tmp . (tmp )) imageFiles ocr1 :: String -> FilePath -> FilePath -> IO T.Text ocr1 language tmp input = T.decodeUtf8 . LB.toStrict -- XXX `--oem 1` seems to be unavailable <$> sh (printf "tesseract '%s' - -l '%s' --oem 3 --psm 1" (tmp input) language) data PdfInfo = PdfInfo { numPages :: Int, pageSize :: (Double, Double) } deriving (Show) data PdfInfoException = PdfInfoException deriving (Show) instance Exception PdfInfoException parsePdfInfo :: LB.ByteString -> PdfInfo parsePdfInfo out' = fromMaybe (throw PdfInfoException) $ do numPages <- readMaybe . T.unpack =<< M.lookup "Pages" kvs pageSize <- rightToMaybe . A.parseOnly pageSizeParser =<< M.lookup "Page size" kvs pure PdfInfo {..} where out = T.decodeUtf8 (LB.toStrict out') kvs = M.fromList . map (second T.stripStart) . map (second T.tail . T.break (== ':')) . filter (not . T.null) . T.lines $ out pageSizeParser = (,) <$> (A.double <* A.string " x ") <*> (A.double <* A.string " pts (A4)") <* A.endOfInput type PdfImages = [PdfImage] data PdfImage = PdfImage { page :: Int, num :: Int, type_ :: String, width :: Int, height :: Int, color :: String, comp :: Int, bpc :: Int, enc :: String, interp :: String, object :: Int, id :: Int, xPpi :: Int, yPpi :: Int, size :: String, ratio :: String } deriving (Show) imageSize :: PdfImage -> (Double, Double) imageSize (PdfImage {..}) = let f ppi p = 72 * fromIntegral p / fromIntegral ppi in (f xPpi width, f yPpi height) data PdfImagesException = PdfImagesException deriving (Show) instance Exception PdfImagesException data ProcessException = ProcessException Int LB.ByteString deriving (Show) instance Exception ProcessException parsePdfImages :: LB.ByteString -> PdfImages parsePdfImages out' = map ( \(page' : num' : type_ : width' : height' : color : comp' : bpc' : enc : interp : object' : id' : xPpi' : yPpi' : size : ratio : []) -> PdfImage { page = read page', num = read num', width = read width', height = read height', comp = read comp', bpc = read bpc', object = read object', id = read id', xPpi = read xPpi', yPpi = read yPpi', .. } ) . map (map T.unpack) . map T.words . drop 2 . filter (not . T.null) . T.lines $ out where out = T.decodeUtf8 (LB.toStrict out') sh :: String -> IO LB.ByteString sh cmd = do -- printf "+ %s\n" cmd (exitCode, out, err) <- readProcess (fromString cmd) case exitCode of ExitSuccess -> return out ExitFailure exitCode' -> throwIO $ ProcessException exitCode' err sh_ :: String -> IO () sh_ cmd = do -- printf "+ %s\n" cmd (exitCode, err) <- readProcessStderr (fromString cmd) case exitCode of ExitSuccess -> return () ExitFailure exitCode' -> throwIO $ ProcessException exitCode' err rightToMaybe :: Either e a -> Maybe a rightToMaybe = either (const Nothing) Just parMapM :: (a -> IO b) -> [a] -> IO [b] parMapM f xs = do n <- getNumProcessors withPool n $ \pool -> parallel pool (map f xs) parMapM_ :: (a -> IO b) -> [a] -> IO () parMapM_ f = fmap (const ()) . parMapM f