bisc/Main.hs

141 lines
3.9 KiB
Haskell
Raw Permalink Normal View History

2018-09-21 21:34:31 +02:00
{-# LANGUAGE DeriveGeneric #-}
{-# LANGUAGE OverloadedStrings #-}
{-# LANGUAGE OverloadedLabels #-}
{-# LANGUAGE FlexibleContexts #-}
2019-03-25 00:02:38 +01:00
import Data.List (nub)
import Data.Maybe (mapMaybe)
import Data.Configurator
import Control.Monad (mapM_, filterM)
import Control.Monad.Reader (ReaderT, runReaderT, asks)
import System.FilePath (joinPath, takeBaseName, (</>))
import System.IO (readFile)
2019-06-02 10:39:07 +02:00
import System.Directory
2019-03-25 00:02:38 +01:00
2018-09-21 21:34:31 +02:00
import Database.Selda
import Database.Selda.SQLite
2019-03-25 00:02:38 +01:00
2019-06-02 10:39:07 +02:00
import qualified Data.Text as T
import qualified Data.Text.IO as T
2019-03-25 00:02:38 +01:00
data Settings = Settings
{ whitelistPath :: FilePath
, webenginePath :: FilePath
}
2018-09-21 21:34:31 +02:00
data Cookie = Cookie
{ host_key :: Text
, creation_utc :: Int
} deriving (Generic, Show)
instance SqlRow Cookie
2018-09-23 13:40:12 +02:00
2019-03-25 00:02:38 +01:00
type Action = ReaderT Settings IO
2018-09-21 21:34:31 +02:00
2018-09-23 13:40:12 +02:00
2019-03-25 00:02:38 +01:00
main :: IO ()
main = do
2019-06-02 10:39:07 +02:00
config <- getXdgDirectory XdgConfig ("bisc" </> "bisc.conf")
2019-03-25 00:02:38 +01:00
settings <- loadSettings config
runReaderT clean settings
2018-09-21 21:34:31 +02:00
2019-03-25 00:02:38 +01:00
clean :: Action ()
clean = do
path <- asks whitelistPath
whitelist <- liftIO (T.lines <$> T.readFile path)
(n, bad) <- deleteCookies whitelist
if (n > 0)
then do
log ("Cookies: deleted " <> num n <> " from:")
log (prettyPrint bad)
else log ("Cookies: nothing to delete.")
2018-09-21 21:34:31 +02:00
2019-03-25 00:02:38 +01:00
(n, bad) <- deleteData whitelist
if (n > 0)
then do
log ("Persistent data: deleted " <> num n <> " entries:")
log (prettyPrint bad)
else log ("Persistent data: nothing to delete.")
2018-09-21 21:34:31 +02:00
2019-03-25 00:02:38 +01:00
where log = liftIO . T.putStrLn
num = T.pack . show
2018-09-23 13:40:12 +02:00
2019-03-25 00:02:38 +01:00
deleteCookies :: [Text] -> Action (Int, [Text])
2018-09-23 13:40:12 +02:00
deleteCookies domains = do
2019-03-25 00:02:38 +01:00
database <- (</> "Cookies") <$> asks webenginePath
liftIO $ withSQLite database $ do
2018-09-21 21:34:31 +02:00
bad <- query $ do
cookie <- select cookies
restrict (by whitelist cookie)
return (cookie ! #host_key)
n <- deleteFrom cookies (by whitelist)
2018-09-23 13:40:12 +02:00
return (n, nub bad)
where
by set x = not_ (x ! #host_key `isIn` set)
whitelist = map text domains
2019-03-25 00:02:38 +01:00
deleteData :: [Text] -> Action (Int, [Text])
deleteData whitelist = do
webengine <- asks webenginePath
appCache <- liftIO $ listDirectoryAbs (webengine </> "Application Cache")
indexedDB <- liftIO $ listDirectoryAbs (webengine </> "IndexedDB")
localStorage <- liftIO $ listDirectoryAbs (webengine </> "Local Storage")
2018-09-23 13:40:12 +02:00
let
2019-03-25 00:02:38 +01:00
entries = appCache ++ indexedDB ++ localStorage
2018-09-23 13:40:12 +02:00
badFiles = filterMaybe (fmap unlisted . domain) entries
badDomains = mapMaybe domain badFiles
liftIO $ mapM_ removePathForcibly badFiles
2018-09-23 13:40:12 +02:00
return (length badFiles, nub badDomains)
where
listDirectoryAbs :: FilePath -> IO [FilePath]
listDirectoryAbs dir = map (dir </>) <$> listDirectory dir
2018-09-23 13:40:12 +02:00
maybeToBool :: Maybe Bool -> Bool
maybeToBool Nothing = False
maybeToBool (Just x) = x
filterMaybe :: (a -> Maybe Bool) -> [a] -> [a]
filterMaybe f = filter (maybeToBool . f)
domain :: FilePath -> Maybe Text
domain = extract . url where
extract [] = Nothing
extract (x:[]) = Nothing
extract (x:xs) = Just $ T.unwords (init xs)
url = T.splitOn "_" . T.pack . takeBaseName
unlisted = not . (`elem` whitelist)
2019-03-25 00:02:38 +01:00
loadSettings :: FilePath -> IO Settings
loadSettings path = do
2019-06-02 10:39:07 +02:00
configdir <- getXdgDirectory XdgConfig "qutebrowser"
datadir <- getXdgDirectory XdgData "qutebrowser"
2019-03-25 00:02:38 +01:00
let
defaultWhitelist = joinPath [configdir, "whitelists", "cookies"]
defaultWebengine = joinPath [datadir, "webengine"]
2018-09-21 21:34:31 +02:00
2019-03-25 00:02:38 +01:00
config <- load [Optional path]
whitelist <- lookupDefault defaultWhitelist config "whitelist-path"
webengine <- lookupDefault defaultWebengine config "webengine-path"
return (Settings whitelist webengine)
2018-09-21 21:34:31 +02:00
2019-03-25 00:02:38 +01:00
prettyPrint :: [Text] -> Text
prettyPrint = T.unlines . bullet
where bullet = map (" * " <>)
getDirectoryFiles :: FilePath -> IO [FilePath]
getDirectoryFiles path = map (path </>) <$>
getDirectoryContents path >>= filterM doesFileExist
cookies :: Table Cookie
cookies = table "cookies" []