1
1
mirror of https://github.com/aelve/guide.git synced 2024-11-29 14:35:35 +03:00

Added the parsing of lts list from stackage.org

This commit is contained in:
Boris M. Yartsev 2017-06-11 16:38:52 +02:00
parent f588861523
commit b257c44e36
15 changed files with 509 additions and 336 deletions

View File

@ -3,18 +3,8 @@
module Main where module Main where
import qualified Data.ByteString.Lazy as BL
import Data.Int(Int64)
import qualified Control.Exception as X
import qualified Data.Char as DC
import qualified Data.List as DL
import Control.Monad(forever)
import System.Directory(copyFile)
import System.IO (stdout, hFlush)
import qualified Data.Map.Strict as Map
import Data.Default import Data.Default
import IndexProject import IndexProject
main :: IO () main :: IO ()
main = processCycle def main = processREPLCycle def

View File

@ -15,7 +15,9 @@ cabal-version: >=1.10
library library
hs-source-dirs: src hs-source-dirs: src
other-modules: Common, FileUtils, HttpDownload, HackageUpdate, REPL, Storage other-modules: Common, FileUtils, HttpDownload, HackageUpdate, REPL
, HackageCommands, StackageUpdate, StackageCommands
exposed-modules: IndexProject, HackageArchive, Stackage exposed-modules: IndexProject, HackageArchive, Stackage
build-depends: base >= 4.7 && < 5 build-depends: base >= 4.7 && < 5
, directory , directory
@ -42,6 +44,7 @@ library
, transformers , transformers
, zlib , zlib
, acid-state , acid-state
, unordered-containers
default-language: Haskell2010 default-language: Haskell2010

View File

@ -1,15 +1,27 @@
module Common(URL, module Common(URL,
PackageName, PackageName,
PackageVersion(..),
PackageData, PackageData,
SnapshotData(..), SnapshotData(..),
UpdateArchiveException(..), UpdateArchiveException(..),
UpdateInfo(..),
HackageUpdateInfo(..), HackageUpdateInfo(..),
getArchive, getArchive,
getArchiveClone, getArchiveClone,
getTar, getTar,
getTarClone, getTarClone,
parseIntEnd, parseIntEnd,
parseValEnd) where parseValEnd,
ShortSnapshotName,
LongSnapshotName,
shortName,
longName,
getSnapshotURL,
StackageSnapshot,
StackageSnapshots(..),
StackageLTS,
StackageUpdateInfo(..)) where
import qualified Control.Exception as X import qualified Control.Exception as X
import qualified Data.ByteString.Lazy as BL import qualified Data.ByteString.Lazy as BL
@ -23,7 +35,22 @@ import System.FilePath((</>))
type URL = String type URL = String
type PackageName = String type PackageName = String
type PackageData = (PackageName, DV.Version) data PackageVersion = Installed | Specified DV.Version deriving (Eq, Ord, Show)
type PackageData = (PackageName, PackageVersion)
data UpdateInfo = UI {
iuh :: HackageUpdateInfo,
sui :: StackageUpdateInfo
} deriving (Eq, Show)
instance Default UpdateInfo where
def = defaultUI
defaultUI :: UpdateInfo
defaultUI = UI {
iuh = defaultIUH,
sui = defaultSUI
}
data SnapshotData = SnapshotData { data SnapshotData = SnapshotData {
md5Hash :: String, md5Hash :: String,
@ -42,6 +69,7 @@ data HackageUpdateInfo = IUH {
iuhArchiveURL :: URL iuhArchiveURL :: URL
} deriving (Eq, Show) } deriving (Eq, Show)
instance Default HackageUpdateInfo where instance Default HackageUpdateInfo where
def = defaultIUH def = defaultIUH
@ -86,3 +114,35 @@ parseValEnd :: String -> String
parseValEnd val | DL.length l > 1 = DL.last l parseValEnd val | DL.length l > 1 = DL.last l
| otherwise = "" | otherwise = ""
where l = words val where l = words val
-- Stackage stuff
type ShortSnapshotName = String
type LongSnapshotName = String
type StackageSnapshot = (ShortSnapshotName, LongSnapshotName)
newtype StackageSnapshots = SSS [StackageSnapshot] deriving (Eq, Show)
shortName :: StackageSnapshot -> String
shortName = fst
longName :: StackageSnapshot -> String
longName = snd
type StackageLTS = (LongSnapshotName, [PackageData])
getLTSURL :: StackageUpdateInfo -> LongSnapshotName -> URL
getLTSURL sui name = suiStackageURL sui </> name </> "cabal.config"
getSnapshotURL :: StackageUpdateInfo -> URL
getSnapshotURL sui = suiStackageURL sui </> "download/lts-snapshots.json"
data StackageUpdateInfo = SUI {
suiUpdateDir :: FilePath,
suiStackageURL :: URL
} deriving (Eq, Show)
defaultSUI :: StackageUpdateInfo
defaultSUI = SUI {
suiUpdateDir = "stackagefiles",
suiStackageURL = "https://www.stackage.org/"
}

View File

@ -1,9 +1,17 @@
{-# LANGUAGE OverloadedStrings #-} {-# LANGUAGE OverloadedStrings #-}
{-# LANGUAGE TypeFamilies #-}
{-# LANGUAGE DeriveDataTypeable #-}
{-# LANGUAGE TemplateHaskell #-}
module HackageArchive ( module HackageArchive (
buildDifferenceMap, buildDifferenceMap,
buildHackageMap, buildHackageMap,
buildPreHackageMap, buildPreHackageMap,
updatePersistentMap,
printPersistentDiffMap,
queryPersistentMap,
HackagePackage (..), HackagePackage (..),
HackageName, HackageName,
HackageMap, HackageMap,
@ -26,11 +34,18 @@ import qualified Distribution.PackageDescription.Parse as DPDP
import qualified Data.Map.Strict as M import qualified Data.Map.Strict as M
import qualified Control.Exception as X import qualified Control.Exception as X
import Data.Typeable
import Data.Acid
import Data.Acid.Advanced
import Data.SafeCopy
import Control.Monad.Reader
import Data.Maybe import Data.Maybe
import Debug.Trace import Debug.Trace
import Control.Monad(guard) import Control.Monad(guard)
import qualified Data.ByteString.Lazy.UTF8 as UTFC import qualified Data.ByteString.Lazy.UTF8 as UTFC
import qualified Control.Monad.State as State
import System.FilePath.Posix(hasTrailingPathSeparator) import System.FilePath.Posix(hasTrailingPathSeparator)
import Common import Common
@ -40,9 +55,8 @@ type HackageName = String
-- The record for each of the package from hackage -- The record for each of the package from hackage
-- TODO - add another information about the packages -- TODO - add another information about the packages
data HackagePackage = HP { data HackagePackage = HP {
-- packageData :: HHPathData
name :: HackageName, name :: HackageName,
version :: DV.Version, pVersion :: DV.Version,
author :: String author :: String
} deriving (Eq, Show) } deriving (Eq, Show)
@ -52,8 +66,7 @@ data HackageUpdate = Added | Removed | Updated deriving (Eq, Show)
-- The map of all the hackage packages with name as the key and HackagePackage -- The map of all the hackage packages with name as the key and HackagePackage
-- as the value -- as the value
type HackageMap = M.Map HackageName HackagePackage type HackageMap = M.Map HackageName HackagePackage
type PreHackageMap = M.Map HackageName PackageVersion
type PreHackageMap = M.Map HackageName DV.Version
-- The map, that shows, which packages have change since the last update -- The map, that shows, which packages have change since the last update
type HackageUpdateMap = M.Map HackageName (HackageUpdate, HackagePackage) type HackageUpdateMap = M.Map HackageName (HackageUpdate, HackagePackage)
@ -69,7 +82,7 @@ parseCabalFilePath = do
guard (name == package) guard (name == package)
suff <- RP.string ".cabal" suff <- RP.string ".cabal"
RP.eof RP.eof
pure (package, version) pure (package, Specified version)
where phi l = DC.isLetter l || l == '-' where phi l = DC.isLetter l || l == '-'
updateMapCompare :: (Ord a) => String -> a -> M.Map String a -> M.Map String a updateMapCompare :: (Ord a) => String -> a -> M.Map String a -> M.Map String a
@ -90,7 +103,7 @@ buildDifferenceMap oldMap newMap = foldr M.union M.empty [deletedMap, addedMap,
diff newpack oldpack = if newpack /= oldpack then Just newpack else Nothing diff newpack oldpack = if newpack /= oldpack then Just newpack else Nothing
createPackage :: DPD.PackageDescription -> HackagePackage createPackage :: DPD.PackageDescription -> HackagePackage
createPackage pd = HP { name = nm, version = ver, author = auth } createPackage pd = HP { name = nm, pVersion = ver, author = auth }
where where
pkg = DPD.package pd pkg = DPD.package pd
nm = DP.unPackageName (DP.pkgName pkg) nm = DP.unPackageName (DP.pkgName pkg)
@ -143,3 +156,56 @@ buildHackageMap Tar.Done _ = M.empty
buildHackageMap (Tar.Fail e) _ = X.throw e buildHackageMap (Tar.Fail e) _ = X.throw e
-- The stuff needed for acid serialization
newtype KeyValue = KeyValue HackageMap deriving (Typeable)
$(deriveSafeCopy 0 'base ''DV.Version)
$(deriveSafeCopy 0 'base ''HackagePackage)
$(deriveSafeCopy 0 'base ''KeyValue)
$(deriveSafeCopy 0 'base ''HackageUpdate)
insertKey :: HackageName -> HackagePackage -> Update KeyValue ()
insertKey key value = do
KeyValue hackageMap <- State.get
State.put (KeyValue (M.insert key value hackageMap))
updateMap :: HackageMap -> Update KeyValue ()
updateMap newMap = State.put (KeyValue newMap)
lookupKey :: HackageName -> Query KeyValue (Maybe HackagePackage)
lookupKey key = do
KeyValue m <- ask
return (M.lookup key m)
compareMap :: HackageMap -> Query KeyValue HackageUpdateMap
compareMap newMap = do
KeyValue oldMap <- ask
return (buildDifferenceMap oldMap newMap)
$(makeAcidic ''KeyValue ['insertKey, 'lookupKey, 'compareMap, 'updateMap])
updatePersistentMap :: FilePath -> HackageMap -> IO ()
updatePersistentMap path newMap = do
acid <- openLocalStateFrom path (KeyValue M.empty)
do
putStrLn "Updating the persistent map"
update acid (UpdateMap newMap)
closeAcidState acid
printPersistentDiffMap :: FilePath -> HackageMap -> IO ()
printPersistentDiffMap path newMap = do
acid <- openLocalStateFrom path (KeyValue M.empty)
do
diffMap <- query acid (CompareMap newMap)
putStrLn "Printing difference map with persistent map"
mapM_ (print.snd) $ M.toList diffMap
closeAcidState acid
queryPersistentMap :: FilePath -> HackageName -> IO (Maybe HackagePackage)
queryPersistentMap path name = do
acid <- openLocalStateFrom path (KeyValue M.empty)
val <- query acid (LookupKey name)
closeAcidState acid
return val

151
REPL/src/HackageCommands.hs Normal file
View File

@ -0,0 +1,151 @@
module HackageCommands(
showTarElements,
showTarPreElements,
showFileSnapshot,
showFileSubstring,
showUpdateData,
copyArchive,
showDiffMap,
cutFile,
unzipArchive,
removeArchiveFiles,
showArchiveCompare,
updateArchive,
updateArchiveVoid,
updateTotalArchive,
updatePersistentFromTar,
showPersistentQuery,
showPersistentTarCompare
) where
import qualified Data.Map.Strict as M
import Data.Int(Int64)
import System.Directory(copyFile)
import Control.Monad(void)
import FileUtils
import Common
import HackageArchive
import HackageUpdate
-- shows the first count elements, parsed from the tar archive
showTarElements :: FilePath -> Int -> IO ()
showTarElements path count = do
putStrLn $ "Displaying " ++ show count ++ " entries for " ++ path
tar <- loadTar path
mapM_ (print.snd) $ take count $ M.toList $ buildHackageMap tar (buildPreHackageMap tar)
-- shows the first count pre elements (only path is parsed) form the tar archive
showTarPreElements :: FilePath -> Int -> IO ()
showTarPreElements path count = do
putStrLn $ "Pre displaying " ++ show count ++ " entries for " ++ path
tar <- loadTar path
mapM_ print $ take count $ M.toList $ buildPreHackageMap tar
-- Displays the snapshot of the file
showFileSnapshot :: FilePath -> IO()
showFileSnapshot file = do
filesnapshot <- calcFileData file
putStrLn $ "File result for " ++ file
putStrLn $ "\tFile snapshot: " ++ show filesnapshot
-- Shows the update data for the archive on disk
showUpdateData :: FilePath -> URL -> IO()
showUpdateData file json = do
(range, snapshot, filesnapshot) <- calcUpdateResultIO file json
putStrLn $ "Update result for file " ++ file
putStrLn $ "\tHackage snapshot: " ++ show snapshot
putStrLn $ "\tFile snapshot: " ++ show filesnapshot
putStrLn $ "\tRange to update: " ++ show range
-- shows the substring of specified length from file from offset
showFileSubstring :: FilePath -> Int64 -> Int64 -> IO ()
showFileSubstring file from length = do
putStrLn $ "Showing " ++ file ++ " substr"
putStr "\t"
substr <- getFileSubstring file from length
print substr
-- Copies the archive from first filename to the second
copyArchive :: FilePath -> FilePath -> IO ()
copyArchive archive1 archive2 = do
copyFile archive1 archive2
putStrLn $ "Copied the " ++ archive1 ++ " to " ++ archive2
-- Shows the difference between two tar archives, by building the pre maps of
-- each of them, and then comparing
showDiffMap :: FilePath -> FilePath -> IO ()
showDiffMap newTarFile oldTarFile = do
putStrLn $ "Displaying difference between " ++ newTarFile ++ " and " ++ oldTarFile
newTar <- loadTar newTarFile
oldTar <- loadTar oldTarFile
let newMap = buildHackageMap newTar (buildPreHackageMap newTar)
let oldMap = buildHackageMap oldTar (buildPreHackageMap oldTar)
let diffMap = buildDifferenceMap oldMap newMap
mapM_ (print.snd) $ M.toList diffMap
-- this method cuts the data from the end of the archive,
-- because hackage 01-index.tar.gz is not strictly incremental
cutFile :: FilePath -> Int64 -> IO()
cutFile path size = do
truncateIfExists path size
putStrLn $ "Cut " ++ show size ++ " bytes from " ++ path
-- Unzips the gz archive to tar
unzipArchive :: FilePath -> FilePath -> IO()
unzipArchive archive tar = do
putStrLn $ "Unzipping " ++ archive ++ " to " ++ tar
unzipFile archive tar
-- Removes gz and tar files
removeArchiveFiles :: FilePath -> FilePath -> IO()
removeArchiveFiles archive tar = do
putStrLn $ "Removing archive files " ++ archive ++ " " ++ tar
removeIfExists archive
removeIfExists tar
-- Compares the two gz archives. Needed to find that the archive was not incremental
showArchiveCompare :: FilePath -> FilePath -> IO()
showArchiveCompare archive1 archive2= do
val <- compareFiles archive1 archive2
putStrLn $ "Compare result " ++ archive1 ++ " " ++ archive2 ++ " " ++ show val
updateArchive :: URL -> URL -> FilePath -> IO UpdateResult
updateArchive = performArchiveFileUpdate
updateArchiveVoid :: URL -> URL -> FilePath -> IO ()
updateArchiveVoid snapshotURL archiveURL archive =
void (performArchiveFileUpdate snapshotURL archiveURL archive)
updateTotalArchive :: IO UpdateResult -> IO() -> IO() -> IO ()
updateTotalArchive update unzip persist = do
putStrLn "Performing total update"
result <- update
if result == ArchiveIsOk then putStrLn "Nothing to update"
else unzip >> persist
updatePersistentFromTar :: FilePath -> FilePath -> IO()
updatePersistentFromTar updateDir tarFile = do
newTar <- loadTar tarFile
let newMap = buildHackageMap newTar (buildPreHackageMap newTar)
updatePersistentMap updateDir newMap
showPersistentQuery :: FilePath -> HackageName -> IO()
showPersistentQuery updateDir name = do
putStrLn $ "Querying storage hackage map with " ++ name
value <- queryPersistentMap updateDir name
case value of
Just package -> do
putStrLn "Found"
print package
Nothing -> putStrLn "Not found"
showPersistentTarCompare :: FilePath -> FilePath -> IO()
showPersistentTarCompare updateDir tarFile = do
newTar <- loadTar tarFile
let newMap = buildHackageMap newTar (buildPreHackageMap newTar)
printPersistentDiffMap updateDir newMap

View File

@ -2,7 +2,8 @@
{-# LANGUAGE ScopedTypeVariables #-} {-# LANGUAGE ScopedTypeVariables #-}
module HackageUpdate ( module HackageUpdate (
performArchiveFileUpdate, performArchiveFileUpdate,
calcUpdateResultIO) where calcUpdateResultIO,
UpdateResult(..)) where
import Data.Aeson.Types import Data.Aeson.Types
import qualified Data.Aeson as A import qualified Data.Aeson as A
@ -67,7 +68,7 @@ calcUpdateResultIO file json = do
parseSnapshotJSONThrow :: BL.ByteString -> IO HackageSnapshotData parseSnapshotJSONThrow :: BL.ByteString -> IO HackageSnapshotData
parseSnapshotJSONThrow body = case A.decode body of parseSnapshotJSONThrow body = case A.decode body of
(Just snapshot) -> return snapshot (Just snapshot) -> return snapshot
Nothing -> X.throwIO $ UAE "Could not decode JSON" Nothing -> X.throwIO $ UAE "Could not decode hackage JSON"
-- Returns the snapshot of archive from the hackage -- Returns the snapshot of archive from the hackage
fetchSnapshot :: URL -> IO HackageSnapshotData fetchSnapshot :: URL -> IO HackageSnapshotData

View File

@ -1,12 +1,12 @@
module IndexProject(HackageUpdateInfo(..), module IndexProject(
HackageName(..), processREPLCycle
processCycle,
updateHackageMap,
queryHackageMap
) where ) where
import REPL (HackageUpdateInfo(..), processCycle, updateArchive, updateMapFromTar, queryHackageMap) import REPL (processREPLCycle)
import HackageArchive (HackageName (..), HackagePackage(..)) --import HackageArchive (HackageName (..), HackagePackage(..))
updateHackageMap :: HackageUpdateInfo -> IO ()
updateHackageMap iuh = updateArchive iuh >> updateMapFromTar iuh --HackageUpdateInfo(..),
-- HackageName(..),
--updateHackageMap :: HackageUpdateInfo -> IO ()
--updateHackageMap iuh = updateArchive iuh >> updateMapFromTar iuh

View File

@ -1,253 +1,153 @@
{-# LANGUAGE OverloadedStrings #-} {-# LANGUAGE OverloadedStrings #-}
{-# LANGUAGE ScopedTypeVariables #-} {-# LANGUAGE ScopedTypeVariables #-}
module REPL ( processCycle, module REPL ( processREPLCycle ) where
updateArchive,
updateMapFromTar,
queryHackageMap,
HackageUpdateInfo (..)
) where
import qualified Data.Map.Strict as M
import qualified Data.Char as DC import qualified Data.Char as DC
import qualified Control.Exception as X import qualified Control.Exception as X
import Data.List(isPrefixOf)
import Control.Monad(forever, void) import Control.Monad(forever, void)
import System.IO (stdout, hFlush) import System.IO (stdout, hFlush)
import Data.Int(Int64)
import System.Exit(exitSuccess) import System.Exit(exitSuccess)
import System.Directory(copyFile)
import System.FilePath((</>))
import Data.List(isPrefixOf)
import Common import Common
import HackageArchive import qualified HackageCommands as HC
import HackageUpdate import qualified StackageCommands as SC
import FileUtils
import HttpDownload
import Storage
processCycle :: HackageUpdateInfo -> IO () processREPLCycle :: UpdateInfo -> IO ()
processCycle iuh = forever $ do processREPLCycle ui = forever $ do
putStr "Input command: " putStr "Input command: "
hFlush stdout hFlush stdout
command <- getLine command <- getLine
hFlush stdout hFlush stdout
processCommand command `X.catch` eh `X.catch` eh2 `X.catch` eh3 processCommand command `X.catch` eh `X.catch` eh2 `X.catch` eh3
where where
processCommand = buildCommand iuh processCommand = buildCommand ui
eh (e :: X.IOException) = putStrLn $ "IO Error: " ++ show e eh (e :: X.IOException) = putStrLn $ "IO Error: " ++ show e
eh2 (e :: UpdateArchiveException) = putStrLn $ "Parsing error: " ++ show e eh2 (e :: UpdateArchiveException) = putStrLn $ "Parsing error: " ++ show e
eh3 (e :: X.ErrorCall) = putStrLn $ "Error call: " ++ show e eh3 (e :: X.ErrorCall) = putStrLn $ "Error call: " ++ show e
buildCommand :: HackageUpdateInfo -> (String -> IO()) buildCommand :: UpdateInfo -> (String -> IO())
buildCommand iuh = processCommand buildCommand ui = processCommand
where where
processCommand command processCommand command
-- checks the current gzip archive and understands what to download -- checks the current hackage gzip archive and understands what to download
| chk "checkclone" = showUpdateData archC snapURL | chk "check" = HC.showUpdateData arch snapURL
-- checks the current gzip archive and understands what to download -- updates the gzip archive file, unpacks it to tar and loads in the permanent storage
| chk "check" = showUpdateData arch snapURL | chk "totalupdate" = HC.updateTotalArchive updateCommand unzipCommand persistCommand
-- updates the gzip archive file from hackage
| chk "update" = HC.updateArchiveVoid snapURL archURL arch
-- shows the snapshot of hackage gzip archive file (md5 and length)
| chk "file" = HC.showFileSnapshot arch
-- cuts the end of the hackage gzip archive file for checking purposes
| chk "cut" = HC.cutFile arch (parseIntEnd command)
-- unzips the downloaded gzip archive to tar file
| chk "unzip" = HC.unzipArchive arch trFile
-- removes the gzip and tar files
| chk "clean" = HC.removeArchiveFiles arch trFile
-- shows the first 50 pre elements from tar archive
| chk "tarshowpre" = HC.showTarPreElements trFile 50
-- shows the first 50 elements from tar archive
| chk "tarshow" = HC.showTarElements trFile 50
-- Updates the persistent map from tar archive
| chk "tarpersist" = persistCommand
-- compares the map from tar archive and the persistent map
| chk "cmppersist" = HC.showPersistentTarCompare ud trFile
-- shows the package from the persistent map
| chk "querypersist" = HC.showPersistentQuery ud (parseValEnd command)
| chk "fileclone" = showFileSnapshot archC -- shows the snapshots from stackage
| chk "file" = showFileSnapshot arch -- shows the snapshot of hackage file | chk "snapshots" = SC.showSnapshots snapshotsURL
-- exits the REPL
| chk "copyorig" = copyArchive arch archC -- copies the current archive to the orig place
| chk "cutclone" = cutFile archC (parseIntEnd command)
| chk "cut" = cutFile arch (parseIntEnd command) -- cuts the end of the gzip file for checking purposes
| chk "unzipclone" = unzipArchive archC trFileC -- unzips the downloaded gzip archive
| chk "unzip" = unzipArchive arch trFile -- unzips the downloaded gzip archive
| chk "cleanclone" = removeArchiveFiles archC trFileC
| chk "clean" = removeArchiveFiles arch trFile
| chk "tarparsepreclone" = showPreMap trFileC 50 -- loads the tar clone information in the memory
| chk "tarparsepre" = showPreMap trFile 50 -- loads the tar information in the memory
| chk "tarparseclone" = showMap trFileC 50 -- loads the tar clone information in the memory
| chk "tarparse" = showMap trFile 50 -- loads the tar information in the memory
| chk "compare" = showArchiveCompare arch archC
| chk "update" = void $ performArchiveFileUpdate snapURL archURL arch
| chk "acidcompare" = printAcidCompare ud trFile
| chk "acidupdate" = acidUpdate ud trFile
| chk "acidquery" = showAcidQuery ud (parseValEnd command)
| chk "tarcmp" = showDiffMap trFile trFileC
| chk "exit" = exitREPL | chk "exit" = exitREPL
| chk "quit" = exitREPL
| chk "help" = showHelp iuh -- shows the help for REPL commands
| otherwise = showHelp iuh | chk "help" = showHelp ui
-- these are the clones of the commands above for the orig files
-- You'll probably won't need them, unless you are me (borboss366)
-- copies the current hackage archive to other file. Needed for checking properties
| chk "system-copyorig" = HC.copyArchive arch archC
| chk "system-checkclone" = HC.showUpdateData archC snapURL
| chk "system-fileclone" = HC.showFileSnapshot archC
| chk "system-cutclone" = HC.cutFile archC (parseIntEnd command)
| chk "system-unzipclone" = HC.unzipArchive archC trFileC
| chk "system-cleanclone" = HC.removeArchiveFiles archC trFileC
| chk "system-tarshowpreclone" = HC.showTarPreElements trFileC 50
| chk "system-tarshowclone" = HC.showTarElements trFileC 50
-- compares the gzip archive with orig archive, that was copied some time before
| chk "system-compare" = HC.showArchiveCompare arch archC
-- shows diff map between tar and tar.orig archives
| chk "system-tarcmp" = HC.showDiffMap trFile trFileC
| otherwise = showHelp ui
where pc = map DC.toLower command where pc = map DC.toLower command
chk val = val `isPrefixOf` pc chk val = val `isPrefixOf` pc
arch = getArchive iuh arch = (getArchive.iuh) ui
archC = getArchiveClone iuh archC = (getArchiveClone.iuh) ui
archURL = iuhArchiveURL iuh archURL = (iuhArchiveURL.iuh) ui
snapURL = iuhSnapshotURL iuh snapURL = (iuhSnapshotURL.iuh) ui
trFile = getTar iuh trFile = (getTar.iuh) ui
trFileC = getTarClone iuh trFileC = (getTarClone.iuh) ui
ud = iuhUpdateDir iuh ud = (iuhUpdateDir.iuh) ui
showHelp :: HackageUpdateInfo -> IO() snapshotsURL = (getSnapshotURL.sui) ui
showHelp iuh = do
updateCommand = HC.updateArchive snapURL archURL arch
unzipCommand = HC.unzipArchive arch trFile
persistCommand = HC.updatePersistentFromTar ud trFile
showHelp :: UpdateInfo -> IO()
showHelp ui = do
putStrLn "Available commands: " putStrLn "Available commands: "
putStrLn $ "check - downloads the json length and md5 hash from " ++ snapURL ++ putStrLn $ "check - downloads the json length and md5 hash from " ++ snapURL ++
", and compares it with local " ++ arch ", and compares it with local " ++ arch
putStrLn $ "checkclone - same for " ++ archC
putStrLn $ "file - displays the current " ++ arch ++ " length and md5 hash" putStrLn $ "file - displays the current " ++ arch ++ " length and md5 hash"
putStrLn $ "fileclone - same for " ++ archC ++ " file"
putStrLn $ "copyorig - copy the " ++ arch ++ " to " ++ archC
putStrLn $ "cut size - cuts the size bytes from the end of the " ++ arch ++ " , for update command" putStrLn $ "cut size - cuts the size bytes from the end of the " ++ arch ++ " , for update command"
putStrLn "cutclone size - cuts the size bytes from the end of the 01-index.tar.gz, for update command"
putStrLn $ "unzip - unzips the " ++ arch ++ " in the " ++ trFile ++ " file" putStrLn $ "unzip - unzips the " ++ arch ++ " in the " ++ trFile ++ " file"
putStrLn $ "unzipclone - unzips the " ++ archC ++ " in the " ++ trFileC ++ " file"
putStrLn $ "clean - deletes the " ++ arch ++ " and " ++ trFile putStrLn $ "clean - deletes the " ++ arch ++ " and " ++ trFile
putStrLn $ "cleanclone - deletes the " ++ archC ++ " and " ++ trFileC
putStrLn $ "compare - compares the " ++ arch ++ " with " ++ archC
putStrLn $ "tarparse - loads the map of entries from " ++ trFile ++ " and displays it"
putStrLn $ "tarparseclone - same for " ++ trFileC
putStrLn $ "tarparsepre - loads the premap of entries from " ++ trFile ++ " and displays it"
putStrLn $ "tarparsepreclone - same for " ++ trFileC
putStrLn $ "tarcmp - compares the entries of " ++ trFile ++ " and " ++ trFileC
putStrLn $ "update - updates the current " ++ arch ++ " from " ++ archURL putStrLn $ "update - updates the current " ++ arch ++ " from " ++ archURL
putStrLn $ "acidcompare - compares the state of " ++ trFile ++ " with map from acid state" putStrLn $ "totalupdate - updates the current " ++ arch ++ " from " ++ archURL
putStrLn $ "tarshow - loads the map of entries from " ++ trFile ++ " and displays it"
putStrLn $ "tarshowpre - loads the premap of entries from " ++ trFile ++ " and displays it"
putStrLn $ "cmppersist - compares the state of " ++ trFile ++ " with map from persistent storage"
putStrLn $ "tarpersist - updates the persistent storage with " ++ trFile
putStrLn "querypersist name - queries the persistent storage with package"
putStrLn $ "snapshots - show the stackage snapshots from " ++ snapshotsURL
putStrLn "exit - exits this repl"
putStrLn "help - shows this help"
{-
putStrLn $ "compare - compares the " ++ arch ++ " with " ++ archC
putStrLn $ "tarcmp - compares the entries of " ++ trFile ++ " and " ++ trFileC
putStrLn $ "acidcmp - compares the state of " ++ trFile ++ " with map from acid state"
putStrLn $ "acidupdate - updates the acid state with " ++ trFile putStrLn $ "acidupdate - updates the acid state with " ++ trFile
putStrLn "acidquery name - queries the acid with package" putStrLn "acidquery name - queries the acid with package"
putStrLn "exit - exits this repl" putStrLn $ "checkclone - same for " ++ archC
putStrLn $ "fileclone - same for " ++ archC ++ " file"
putStrLn $ "copyorig - copy the " ++ arch ++ " to " ++ archC
putStrLn "cutclone size - cuts the size bytes from the end of the 01-index.tar.gz, for update command"
putStrLn $ "unzipclone - unzips the " ++ archC ++ " in the " ++ trFileC ++ " file"
putStrLn $ "cleanclone - deletes the " ++ archC ++ " and " ++ trFileC
-}
where where
arch = getArchive iuh arch = (getArchive.iuh) ui
archC = getArchiveClone iuh archC = (getArchiveClone.iuh) ui
archURL = iuhArchiveURL iuh archURL = (iuhArchiveURL.iuh) ui
snapURL = iuhSnapshotURL iuh snapURL = (iuhSnapshotURL.iuh) ui
trFile = getTar iuh trFile = (getTar.iuh) ui
trFileC = getTarClone iuh trFileC = (getTarClone.iuh) ui
snapshotsURL = (getSnapshotURL.sui) ui
-- Displays the snapshot of the file
showFileSnapshot :: FilePath -> IO()
showFileSnapshot file = do
filesnapshot <- calcFileData file
putStrLn $ "File result for " ++ file
putStrLn $ "\tFile snapshot: " ++ show filesnapshot
-- Shows the update data for the archive on disk
showUpdateData :: FilePath -> URL -> IO()
showUpdateData file json = do
(range, snapshot, filesnapshot) <- calcUpdateResultIO file json
putStrLn $ "Update result for file " ++ file
putStrLn $ "\tHackage snapshot: " ++ show snapshot
putStrLn $ "\tFile snapshot: " ++ show filesnapshot
putStrLn $ "\tRange to update: " ++ show range
-- shows the substring of specified length from file from offset
showFileSubstring :: FilePath -> Int64 -> Int64 -> IO ()
showFileSubstring file from length = do
putStrLn $ "Showing " ++ file ++ " substr"
putStr "\t"
substr <- getFileSubstring file from length
print substr
-- Copies the archive from first filename to the second
copyArchive :: FilePath -> FilePath -> IO ()
copyArchive archive1 archive2 = do
copyFile archive1 archive2
putStrLn $ "Copied the " ++ archive1 ++ " to " ++ archive2
showMap :: FilePath -> Int -> IO ()
showMap path count = do
putStrLn $ "Displaying " ++ show count ++ " entries for " ++ path
tar <- loadTar path
mapM_ (print.snd) $ take count $ M.toList $ buildHackageMap tar (buildPreHackageMap tar)
showPreMap :: FilePath -> Int -> IO ()
showPreMap path count = do
putStrLn $ "Pre displaying " ++ show count ++ " entries for " ++ path
tar <- loadTar path
mapM_ print $ take count $ {-filter ((elem '-').fst) $-} M.toList $ buildPreHackageMap tar
showDiffMap :: FilePath -> FilePath -> IO ()
showDiffMap newTarFile oldTarFile = do
putStrLn $ "Displaying difference between " ++ newTarFile ++ " and " ++ oldTarFile
newTar <- loadTar newTarFile
oldTar <- loadTar oldTarFile
let newMap = buildHackageMap newTar (buildPreHackageMap newTar)
let oldMap = buildHackageMap oldTar (buildPreHackageMap oldTar)
let diffMap = buildDifferenceMap oldMap newMap
mapM_ (print.snd) $ M.toList diffMap
showArchiveCompare :: FilePath -> FilePath -> IO()
showArchiveCompare archive1 archive2= do
val <- compareFiles archive1 archive2
putStrLn $ "Compare result " ++ archive1 ++ " " ++ archive2 ++ " " ++ show val
showAcidQuery :: FilePath -> HackageName -> IO()
showAcidQuery updateDir name = do
putStrLn $ "Querying acid with " ++ name
value <- queryAcidMap updateDir name
case value of
Just package -> do
putStrLn "Found"
print package
Nothing -> putStrLn "Not found"
exitREPL :: IO() exitREPL :: IO()
exitREPL = putStrLn "Finished working with hackage REPL" >> exitSuccess exitREPL = putStrLn "Finished working with hackage REPL" >> exitSuccess
-- this method cuts the data from the end of the archive
-- needed mostly for testing purposes
cutFile :: FilePath -> Int64 -> IO()
cutFile path size = do
truncateIfExists path size
putStrLn $ "Cut " ++ show size ++ " bytes from " ++ path
unzipArchive :: FilePath -> FilePath -> IO()
unzipArchive archive tar = do
putStrLn $ "Unzipping " ++ archive ++ " to " ++ tar
unzipFile archive tar
removeArchiveFiles :: FilePath -> FilePath -> IO()
removeArchiveFiles archive tar = do
putStrLn $ "Removing archive files " ++ archive ++ " " ++ tar
removeIfExists archive
removeIfExists tar
printAcidCompare :: FilePath -> FilePath -> IO()
printAcidCompare updateDir tarFile = do
newTar <- loadTar tarFile
let newMap = buildHackageMap newTar (buildPreHackageMap newTar)
printAcidDiffMap updateDir newMap
acidUpdate :: FilePath -> FilePath -> IO()
acidUpdate updateDir tarFile = do
newTar <- loadTar tarFile
let newMap = buildHackageMap newTar (buildPreHackageMap newTar)
updateAcidMap updateDir newMap
updateArchive :: HackageUpdateInfo -> IO()
updateArchive iuh = void (performArchiveFileUpdate snapURL archURL arch)
where
arch = getArchive iuh
archURL = iuhArchiveURL iuh
snapURL = iuhSnapshotURL iuh
updateMapFromTar :: HackageUpdateInfo -> IO()
updateMapFromTar iuh = acidUpdate (iuhUpdateDir iuh) (getTar iuh)
queryHackageMap :: HackageUpdateInfo -> HackageName -> IO (Maybe HackagePackage)
queryHackageMap iuh = queryAcidMap (iuhUpdateDir iuh)

View File

@ -19,21 +19,11 @@ import Control.Applicative(empty)
import Common import Common
type ConstraintMap = M.Map PackageName PackageData type ConstraintMap = M.Map PackageName PackageData
type ShortSnapshotName = String
type LongSnapshotName = String
type StackageSnapshot = (ShortSnapshotName, LongSnapshotName)
shortName :: StackageSnapshot -> String
shortName = fst
longName :: StackageSnapshot -> String
longName = snd
type StackageLTS = (LongSnapshotName, [PackageData])
parseStackageLTS :: Parser StackageLTS parseStackageLTS :: Parser StackageLTS
parseStackageLTS = do parseStackageLTS = do
ltsName <- parseLTS ltsName <- parseLTS
eol
manyTill anyChar (string "constraints:") manyTill anyChar (string "constraints:")
packages <- many parsePackageLine packages <- many parsePackageLine
pure (ltsName, packages) pure (ltsName, packages)
@ -52,7 +42,7 @@ parsePackageLine = do
version <- parseVersionVer version <- parseVersionVer
many (char ',') many (char ',')
space space
pure (name, version) pure (name, Specified version)
-- unfortunately the cabal.config does not provide versions for several packages -- unfortunately the cabal.config does not provide versions for several packages
-- And writes tehn in form 'binary installed' -- And writes tehn in form 'binary installed'

View File

@ -0,0 +1,11 @@
module StackageCommands(
showSnapshots) where
import Common
import StackageUpdate
showSnapshots :: URL -> IO()
showSnapshots url = do
SSS snapshots <- fetchStackageSnapshots url
putStrLn $ "Showing snapshots from " ++ url
mapM_ (putStrLn.(\s -> "\tSnapshot: " ++ s).show) snapshots

View File

@ -0,0 +1,49 @@
{-# LANGUAGE OverloadedStrings #-}
{-# LANGUAGE ScopedTypeVariables #-}
module StackageUpdate(fetchStackageSnapshots) where
import Data.Traversable
import Data.Aeson.Types
import qualified Data.Aeson as A
import qualified Data.Aeson.Parser as AP
import qualified Data.Text as T
import qualified Control.Exception as X
import qualified Data.ByteString.Lazy as BL
import qualified Data.HashMap.Strict as HM
import Network.HTTP.Client(parseUrlThrow)
import Common
import HttpDownload
instance FromJSON StackageSnapshots where
parseJSON = withObject "snapshots" $ \o ->
-- I have 'o', which is a HashMap.
SSS <$> (for (HM.toList o) $ \(shortName, longNameVal) -> do
longName <- parseJSON longNameVal
return (T.unpack shortName, longName))
-- The method, that raises an exception, if it was not able to parse the
-- snapshot from JSON
parseSnapshotJSONThrow :: BL.ByteString -> IO StackageSnapshots
parseSnapshotJSONThrow body = case A.decode body of
(Just snapshots) -> return snapshots
Nothing -> X.throwIO $ UAE "Could not decode stackage JSON"
fetchStackageSnapshots :: URL -> IO StackageSnapshots
fetchStackageSnapshots url = parseUrlThrow url >>= fetchResponseData >>= parseSnapshotJSONThrow
{-
parseReferers :: Value -> Parser StackageSnapshots
parseReferers = withObject "referers" $ \o ->
-- Now we have 'o', which is a HashMap. We can use HM.toList to turn it
-- into a list of pairs (domain, referer) and then parse each referer:
for (HM.toList o) $ \(domain, referer) -> do
-- accesses :: [(Text, Int)]
accesses <- HM.toList <$> parseJSON referer
-- accesses' :: [(String, Int)]
let accesses' = map (\(page, n) -> (T.unpack page, n)) accesses
return $ Referer {
domain = T.unpack domain,
pathAccesses = accesses' }
-}

View File

@ -1,73 +0,0 @@
{-# LANGUAGE TypeFamilies #-}
{-# LANGUAGE DeriveDataTypeable #-}
{-# LANGUAGE TemplateHaskell #-}
-- This is modified example from AcidState
module Storage (
printAcidDiffMap,
updateAcidMap,
queryAcidMap) where
import Data.Typeable
import Data.Acid
import Data.Acid.Advanced
import Data.SafeCopy
import Control.Monad.Reader
import qualified Data.Map as M
import qualified Control.Monad.State as State
import HackageArchive
import qualified Data.Version as DV
newtype KeyValue = KeyValue HackageMap deriving (Typeable)
$(deriveSafeCopy 0 'base ''DV.Version)
$(deriveSafeCopy 0 'base ''HackagePackage)
$(deriveSafeCopy 0 'base ''KeyValue)
$(deriveSafeCopy 0 'base ''HackageUpdate)
insertKey :: HackageName -> HackagePackage -> Update KeyValue ()
insertKey key value = do
KeyValue hackageMap <- State.get
State.put (KeyValue (M.insert key value hackageMap))
updateMap :: HackageMap -> Update KeyValue ()
updateMap newMap = State.put (KeyValue newMap)
lookupKey :: HackageName -> Query KeyValue (Maybe HackagePackage)
lookupKey key = do
KeyValue m <- ask
return (M.lookup key m)
compareMap :: HackageMap -> Query KeyValue HackageUpdateMap
compareMap newMap = do
KeyValue oldMap <- ask
return (buildDifferenceMap oldMap newMap)
$(makeAcidic ''KeyValue ['insertKey, 'lookupKey, 'compareMap, 'updateMap])
printAcidDiffMap :: FilePath -> HackageMap -> IO ()
printAcidDiffMap path newMap = do
acid <- openLocalStateFrom path (KeyValue M.empty)
do
diffMap <- query acid (CompareMap newMap)
putStrLn "Printing difference map with acid-state"
mapM_ (print.snd) $ M.toList diffMap
closeAcidState acid
updateAcidMap :: FilePath -> HackageMap -> IO ()
updateAcidMap path newMap = do
acid <- openLocalStateFrom path (KeyValue M.empty)
do
putStrLn "Updating the acid map"
update acid (UpdateMap newMap)
closeAcidState acid
queryAcidMap :: FilePath -> HackageName -> IO (Maybe HackagePackage)
queryAcidMap path name = do
acid <- openLocalStateFrom path (KeyValue M.empty)
val <- query acid (LookupKey name)
closeAcidState acid
return val

View File

@ -31,14 +31,6 @@ testPath text val match = testCase (T.unpack ("Parsing " <> expect match <> " \'
parseStackageTests = testGroup "Stackage parsing tests" parseStackageTests = testGroup "Stackage parsing tests"
[ [
{-
testParse parsePackageLine "constraints: abstract-deque ==0.3," True
, testParse parsePackageLine "constraints: abstract-deque ==0.3" True
, testParse parsePackageLine "constraints: abstract-deque ==0." False
, testParse parsePackageLine "constraints: abstract-deque ==" False
, testParse parsePackageLine "constraints: abst3453#$%#ract-deque ==0.3" False
, testParse parsePackageLine "constraints: abstract-deque ==0.3," True
-}
testParse parsePackageLine " ztail ==1.2" True testParse parsePackageLine " ztail ==1.2" True
, testParse parsePackageLine " adjunctions ==4.3," True , testParse parsePackageLine " adjunctions ==4.3," True
, testParse parsePackageLine "ztail ==1.2" True , testParse parsePackageLine "ztail ==1.2" True
@ -57,18 +49,27 @@ parseStackageTests = testGroup "Stackage parsing tests"
parseCabalConfig = testGroup "Cabal config parsing tests" parseCabalConfig = testGroup "Cabal config parsing tests"
[ [
testStackagePackageLines "sometestfile.cnf" testStackagePackageLines parseStackageLTS "sometestfile.cnf"
, testStackagePackageLines parseStackageLTS "sometestfile2.cnf"
, testFileJustParse parseStackageLTS "sometestfile3.cnf" True
] ]
testStackagePackageLines file = testFileParse (testWorkingDir </> file) -- Well this is code duplication. Somehow need to use testParse function here
parseStackageLTS countPackageLines matchWithStackageLTS
testFileJustParse :: Parser a -> FilePath -> Bool -> TestTree
testFileJustParse p file match = testCase ("Testing file: " ++ file) $ do
fileText <- TIO.readFile (testWorkingDir </> file)
assertBool "Failed" (isRight (runParser p "" fileText) == match)
testStackagePackageLines :: Parser StackageLTS -> FilePath -> TestTree
testStackagePackageLines p file = testFileParse (testWorkingDir </> file)
p countPackageLines matchWithStackageLTS
-- refactor isComment
countPackageLines :: T.Text -> Int countPackageLines :: T.Text -> Int
countPackageLines text = length $ filter isComment lns countPackageLines text = length $ filter isPackageLine lns
where lns = T.lines text where lns = T.lines text
isComment ln = not ("--" `T.isInfixOf` ln) isPackageLine ln = not ("--" `T.isInfixOf` ln)
&& (("installed" `T.isInfixOf` ln) || ("==" `T.isInfixOf` ln))
matchWithStackageLTS :: Int -> StackageLTS -> Bool matchWithStackageLTS :: Int -> StackageLTS -> Bool
matchWithStackageLTS count1 stackage = count1 == (length.snd) stackage matchWithStackageLTS count1 stackage = count1 == (length.snd) stackage

View File

@ -0,0 +1,12 @@
-- Stackage snapshot from: http://www.stackage.org/snapshot/lts-2.10
-- Please place this file next to your .cabal file as cabal.config
-- To only use tested packages, uncomment the following line:
-- remote-repo: stackage-lts-2.10:http://www.stackage.org/lts-2.10
constraints: abstract-deque ==0.3,

View File

@ -0,0 +1,12 @@
-- Stackage snapshot from: http://www.stackage.org/snapshot/lts-2.10ZZZ
-- Please place this file next to your .cabal file as cabal.config
-- To only use tested packages, uncomment the following line:
-- remote-repo: stackage-lts-2.10:http://www.stackage.org/lts-2.10
constraints: abstract-deque ==0.3,