module Web.BowerJson where
import Control.Applicative
import Control.Category ((>>>))
import Data.Char
import Data.Map (Map)
import qualified Data.Text as T
import qualified Data.HashMap.Strict as HashMap
import Data.Aeson
import qualified Data.Aeson.Types as Aeson
import Web.BowerJson.Utils
data BowerJson = BowerJson
{ bowerName :: PackageName
, bowerDescription :: Maybe String
, bowerMain :: [FilePath]
, bowerModuleType :: [ModuleType]
, bowerLicence :: [String]
, bowerIgnore :: [String]
, bowerKeywords :: [String]
, bowerAuthors :: [Author]
, bowerHomepage :: Maybe String
, bowerRepository :: Maybe Repository
, bowerDependencies :: Map PackageName VersionRange
, bowerDevDependencies :: Map PackageName VersionRange
, bowerResolutions :: Map PackageName Version
, isPrivate :: Bool
}
deriving (Show, Eq, Ord)
instance FromJSON BowerJson where
parseJSON =
withObject "BowerJson" $ \o ->
BowerJson <$> (o .: "name" >>= parsePackageName)
<*> o .:? "description"
<*> o .:? "main" .!= []
<*> o .:? "moduleType" .!= []
<*> o .:? "licence" .!= []
<*> o .:? "ignore" .!= []
<*> o .:? "keywords" .!= []
<*> o .:? "authors" .!= []
<*> o .:? "homepage"
<*> o .:? "repository"
<*> o `mapWithArbitraryKeys` "dependencies"
<*> o `mapWithArbitraryKeys` "devDependencies"
<*> o `mapWithArbitraryKeys` "resolutions"
<*> o .:? "private" .!= False
where
liftMaybe :: String -> (a -> Maybe b) -> a -> Aeson.Parser b
liftMaybe msg f =
maybe (fail ("unable to parse a value of type: " ++ msg)) return . f
parsePackageName :: String -> Aeson.Parser PackageName
parsePackageName = liftMaybe "PackageName" mkPackageName
mapWithArbitraryKeys o' field =
(o' .:? field .!= Object HashMap.empty)
>>= parseWithArbitraryKeys parsePackageName
newtype PackageName
= PackageName { runPackageName :: String }
deriving (Show, Eq, Ord)
instance FromJSON PackageName where
parseJSON =
withText "PackageName" $ \text ->
case mkPackageName (T.unpack text) of
Just pkgName -> return pkgName
Nothing -> fail ("unable to validate package name: " ++ show text)
mkPackageName :: String -> Maybe PackageName
mkPackageName str
| satisfyAll predicates str = Just (PackageName str)
| otherwise = Nothing
where
dashOrDot = ['-', '.']
satisfyAll ps x = all ($ x) ps
predicates =
[ not . null
, all isAscii
, all (\c -> isLower c || isDigit c || c `elem` dashOrDot)
, headMay >>> isJustAnd (`notElem` dashOrDot)
, lastMay >>> isJustAnd (`notElem` dashOrDot)
]
isJustAnd = maybe False
data ModuleType
= Globals
| AMD
| Node
| ES6
| YUI
deriving (Show, Eq, Ord, Enum)
moduleTypes :: [(String, ModuleType)]
moduleTypes = map (\t -> (map toLower (show t), t)) [Globals .. YUI]
instance FromJSON ModuleType where
parseJSON =
withText "ModuleType" $ \t ->
case lookup (T.unpack t) moduleTypes of
Just t' -> return t'
Nothing -> fail ("invalid module type: " ++ show t)
data Repository = Repository
{ repositoryUrl :: String
, repositoryType :: String
}
deriving (Show, Eq, Ord)
instance FromJSON Repository where
parseJSON =
withObject "Repository" $ \o ->
Repository <$> o .: "url"
<*> o .: "type"
data Author = Author
{ authorName :: String
, authorEmail :: Maybe String
, authorHomepage :: Maybe String
}
deriving (Show, Eq, Ord)
instance FromJSON Author where
parseJSON (Object o) =
Author <$> o .: "name"
<*> o .:? "email"
<*> o .:? "homepage"
parseJSON (String t) =
pure (Author (unwords s2) email homepage)
where
(email, s1) = takeDelim "<" ">" (words (T.unpack t))
(homepage, s2) = takeDelim "(" ")" s1
parseJSON v =
Aeson.typeMismatch "Author" v
newtype Version
= Version { runVersion :: String }
deriving (Show, Eq, Ord)
instance FromJSON Version where
parseJSON =
withText "Version" (pure . Version . T.unpack)
newtype VersionRange
= VersionRange { runVersionRange :: String }
deriving (Show, Eq, Ord)
instance FromJSON VersionRange where
parseJSON =
withText "VersionRange" (pure . VersionRange . T.unpack)