{-# LANGUAGE OverloadedStrings #-} {-# LANGUAGE DeriveDataTypeable #-} {-# LANGUAGE FlexibleInstances #-} {-# LANGUAGE ScopedTypeVariables #-} module Main where import Control.Monad import Control.Monad.Error import Data.Aeson as A import Data.Label.Abstract import Data.Typeable import Data.Data import Data.List import Data.Maybe import qualified Data.ByteString as BS import qualified Data.ByteString.UTF8 as U import Network.Protocol.Uri.Query import Options.Applicative import System.Directory import System.Exit import System.FilePath import System.Environment import System.IO import Text.Printf import Text.PFormat (pformat) import Text.Namefilter (namefilter) import Text.Show.Pretty as PP import Web.VKHS import Web.VKHS.Curl import Web.VKHS.API.Aeson as A import Web.VKHS.API as STR data Options = Options { verb :: Verbosity , cmdOpts :: CmdOptions } deriving(Show) data CmdOptions = Login LoginOptions | Call CallOptions | Music MusicOptions | UserQ UserOptions | WallQ WallOptions deriving(Show) data LoginOptions = LoginOptions { appid :: String , username :: String , password :: String } deriving(Show) data CallOptions = CO { accessToken :: String , parse :: Bool , method :: String , args :: String } deriving(Show) data MusicOptions = MO { accessToken_m :: String , list_music :: Bool , search_string :: String , name_format :: String , output_format :: String , out_dir :: String , records_id :: [String] } deriving(Show) data UserOptions = UO { accessToken_u :: String , queryString :: String } deriving(Show) data WallOptions = WO { accessToken_w :: String , oid :: String } deriving(Show) loginOptions :: Parser CmdOptions loginOptions = Login <$> (LoginOptions <$> strOption (metavar "APPID" & short 'a' & value "3128877" & help "Application ID, defaults to VKHS" ) <*> argument str (metavar "USER" & help "User name or email") <*> argument str (metavar "PASS" & help "User password")) opts m = let access_token_flag = strOption (short 'a' & m & metavar "ACCESS_TOKEN" & help "Access token. Honores VKQ_ACCESS_TOKEN environment variable") in Options <$> flag Normal Debug (long "verbose" & help "Be verbose") <*> subparser ( command "login" (info loginOptions ( progDesc "Login and print access token (also prints user_id and expiriration time)" )) & command "call" (info (Call <$> (CO <$> access_token_flag <*> switch (long "preparse" & short 'p' & help "Preparse into Aeson format") <*> argument str (metavar "METHOD" & help "Method name") <*> argument str (metavar "PARAMS" & help "Method arguments, KEY=VALUE[,KEY2=VALUE2[,,,]]"))) ( progDesc "Call VK API method" )) & command "music" (info ( Music <$> (MO <$> access_token_flag <*> switch (long "list" & short 'l' & help "List music files") <*> strOption ( metavar "STR" & long "query" & short 'q' & value [] & help "Query string") <*> strOption ( metavar "FORMAT" & short 'f' & value "%o_%i %u\t%t" & help "Listing format, supported tags: %i %o %a %t %d %u" ) <*> strOption ( metavar "FORMAT" & short 'F' & value "%a - %t" & help "FileName format, supported tags: %i %o %a %t %d %u" ) <*> strOption (metavar "DIR" & short 'o' & help "Output directory" & value "") <*> arguments str (metavar "RECORD_ID" & help "Download records") )) ( progDesc "List or download music files")) & command "user" (info ( UserQ <$> (UO <$> access_token_flag <*> strOption (long "query" & short 'q' & help "String to query") )) ( progDesc "Extract various user information")) & command "wall" (info ( WallQ <$> (WO <$> access_token_flag <*> strOption (long "id" & short 'i' & help "Owner id") )) ( progDesc "Extract wall information")) ) api_ :: (A.FromJSON a) => Env CallEnv -> String -> [(String,String)] -> IO a api_ a b c = do r <- A.api' a b c case r of Right x -> return x Left e -> hPutStrLn stderr e >> exitFailure api_str :: Env CallEnv -> String -> [(String,String)] -> IO String api_str a b c = do r <- STR.api a b c case r of Right x -> return (U.toString x) Left e -> hPutStrLn stderr e >> exitFailure mr_format :: String -> MusicRecord -> String mr_format s mr = pformat '%' [ ('i', show . aid) , ('o', show . owner_id) , ('a', namefilter . artist) , ('t', namefilter . title) , ('d', show . duration) , ('u', url) ] s mr ifeither e fl fr = either fl fr e errexit e = hPutStrLn stderr e >> exitFailure checkRight (Left e) = hPutStrLn stderr e >> exitFailure checkRight (Right a) = return a main :: IO () main = do m <- maybe (idm) (value) <$> lookupEnv "VKQ_ACCESS_TOKEN" execParser (info (opts m) idm) >>= cmd cmd :: Options -> IO () -- Login cmd (Options v (Login (LoginOptions a u p))) = do let e = (env a u p allAccess) { verbose = v } ea <- login e ifeither ea errexit $ \(at,uid,expin) -> do printf "%s %s %s\n" at uid expin -- Call user-specified API cmd (Options v (Call (CO act pp mn args))) = do let e = (envcall act) { verbose = v } let s = fw (keyValues "," "=") args case pp of False -> do ea <- api_str e mn s putStrLn (show ea) True -> do (ea :: A.Value) <- api_ e mn s putStrLn $ PP.ppShow ea -- Query audio files cmd (Options v (Music mo@(MO act _ q@(_:_) fmt _ _ _))) = do let e = (envcall act) { verbose = v } Response (SL len ms) <- api_ e "audio.search" [("q",q)] forM_ ms $ \m -> do printf "%s\n" (mr_format fmt m) printf "total %d\n" len -- List audio files cmd (Options v (Music mo@(MO act True [] fmt _ _ _))) = do let e = (envcall act) { verbose = v } (Response ms) <- api_ e "audio.get" [] forM_ ms $ \m -> do printf "%s\n" (mr_format fmt m) cmd (Options v (Music mo@(MO act False [] _ ofmt odir []))) = do errexit "Music record ID is not specified (see --help)" -- Download audio files cmd (Options v (Music mo@(MO act False [] _ ofmt odir rid))) = do let e = (envcall act) { verbose = v } Response ms <- api_ e "audio.getById" [("audios", concat $ intersperse "," rid)] forM_ ms $ \m -> do (fp, h) <- openFileMR odir ofmt m r <- vk_curl_file e (url m) $ \ bs -> do BS.hPut h bs checkRight r printf "%d_%d\n" (owner_id m) (aid m) printf "%s\n" (title m) printf "%s\n" fp cmd (Options v (UserQ uo@(UO act qs))) = do let e = (envcall act) { verbose = v } print qs ea <- A.api e "users.search" [("q",qs),("fields","uid,first_name,last_name,photo,education")] putStrLn $ show ea -- ae <- checkRight ea -- processUQ uo ae -- List wall messages cmd (Options v (WallQ mo@(WO act wid))) = do let e = (envcall act) { verbose = v } (Response (SL len ws)) <- api_ e "wall.get" [("owner_id",wid)] forM_ ws $ \w -> do putStrLn (show $ wdate w) putStrLn (wtext w) printf "total %d\n" len type NameFormat = String openFileMR :: FilePath -> NameFormat -> MusicRecord -> IO (FilePath, Handle) openFileMR [] _ m = do let (_,ext) = splitExtension (url m) temp <- getTemporaryDirectory (fp,h) <- openBinaryTempFile temp ("vkqmusic"++ext) return (fp,h) openFileMR dir fmt m = do let (_,ext) = splitExtension (url m) let name = mr_format fmt m let name' = replaceExtension name ext let fp = (dir name') h <- openBinaryFile fp WriteMode return (fp,h) -- data Collection a = MC { -- response :: [a] -- } deriving (Show,Data,Typeable) -- processMC :: MusicOptions -> Collection MusicRecord -> IO () -- processMC (MO _ _ _) (MC r) = do -- forM_ r $ \m -> do -- printf "%d_%d|%s|%s|%s\n" (owner_id m) (aid m) (artist m) (title m) (url m) -- processMC (MO _ rid False) (MC r) = do -- print r -- parseUsers :: JSValue -> Maybe [JSValue] -- parseUsers (JSObject (JSONObject [("response",(JSArray a))])) = Just a -- parseUsers _ = Nothing -- processUQ :: UserOptions -> JSValue -> IO () -- processUQ (UO _ _) j = do -- let (Just u) = parseUsers j -- a <- fromJS (u !! 1) -- print $ show (a :: UserRecord) -- print $ show $ j