module PGFService(cgiMain,cgiMain',getPath, logFile,stderrToFile, newPGFCache) where import PGF (PGF) import qualified PGF import Cache import FastCGIUtils import URLEncoding import Network.CGI import Text.JSON import Text.PrettyPrint as PP(render, text, (<+>)) import qualified Codec.Binary.UTF8.String as UTF8 (decodeString) import qualified Data.ByteString.Lazy as BS import Control.Concurrent import qualified Control.Exception as E import Control.Monad import Control.Monad.State(State,evalState,get,put) import Data.Char import Data.Function (on) import Data.List (sortBy,intersperse,mapAccumL,nub,isSuffixOf) import qualified Data.Map as Map import Data.Maybe import System.Random import System.Process import System.Exit import System.IO import System.Directory(removeFile) import Fold(fold) -- transfer function for OpenMath LaTeX catchIOE :: IO a -> (E.IOException -> IO a) -> IO a catchIOE = E.catch logFile :: FilePath logFile = "pgf-error.log" newPGFCache = newCache PGF.readPGF getPath = do path <- getVarWithDefault "PATH_TRANSLATED" "" -- apache mod_fastcgi if null path then getVarWithDefault "SCRIPT_FILENAME" "" -- lighttpd else return path cgiMain :: Cache PGF -> CGI CGIResult cgiMain cache = handleErrors . handleCGIErrors $ cgiMain' cache =<< getPath cgiMain' :: Cache PGF -> FilePath -> CGI CGIResult cgiMain' cache path = do command <- liftM (maybe "grammar" (urlDecodeUnicode . UTF8.decodeString)) (getInput "command") case command of "download" -> outputBinary =<< liftIO (BS.readFile path) _ -> pgfMain command =<< liftIO (readCache cache path) pgfMain :: String -> PGF -> CGI CGIResult pgfMain command pgf = case command of "parse" -> out =<< doParse pgf # text % cat % from % limit % trie "complete" -> out =<< doComplete pgf # text % cat % from % limit "linearize" -> out =<< doLinearize pgf # tree % to "linearizeAll" -> out =<< doLinearizes pgf # tree % to "linearizeTable" -> out =<< doLinearizeTabular pgf # tree % to "random" -> cat >>= \c -> depth >>= \dp -> limit >>= \l -> to >>= \to -> liftIO (doRandom pgf c dp l to) >>= out "generate" -> out =<< doGenerate pgf # cat % depth % limit % to "translate" -> out =<< doTranslate pgf # text % cat % from % to % limit % trie "translategroup" -> out =<< doTranslateGroup pgf # text % cat % from % to % limit "grammar" -> out =<< doGrammar pgf # requestAcceptLanguage "abstrtree" -> outputGraphviz =<< abstrTree pgf # graphvizOptions % tree "alignment" -> outputGraphviz =<< alignment pgf # tree % to "parsetree" -> do t <- tree Just l <- from opts <- graphvizOptions outputGraphviz (parseTree pgf l opts t) "abstrjson" -> out . jsonExpr =<< tree "browse" -> join $ doBrowse pgf # optId % cssClass % href % format "html" % getIncludePrintNames "external" -> do cmd <- getInput "external" input <- text doExternal cmd input _ -> throwCGIError 400 "Unknown command" ["Unknown command: " ++ show command] where out = outputJSONP text :: CGI String text = liftM (maybe "" (urlDecodeUnicode . UTF8.decodeString)) $ getInput "input" tree :: CGI PGF.Tree tree = do ms <- getInput "tree" s <- maybe (throwCGIError 400 "No tree given" ["No tree given"]) return ms t <- maybe (throwCGIError 400 "Bad tree" ["tree: " ++ s]) return (PGF.readExpr s) t <- either (\err -> throwCGIError 400 "Type incorrect tree" ["tree: " ++ PGF.showExpr [] t ,render (PP.text "error:" <+> PGF.ppTcError err) ]) (return . fst) (PGF.inferExpr pgf t) return t cat :: CGI (Maybe PGF.Type) cat = do mcat <- getInput "cat" case mcat of Nothing -> return Nothing Just "" -> return Nothing Just cat -> case PGF.readType cat of Nothing -> throwCGIError 400 "Bad category" ["Bad category: " ++ cat] Just typ -> return $ Just typ -- typecheck the category optId :: CGI (Maybe PGF.CId) optId = maybe (return Nothing) rd =<< getInput "id" where rd = maybe err (return . Just) . PGF.readCId err = throwCGIError 400 "Bad identifier" [] cssClass, href :: CGI (Maybe String) cssClass = getInput "css-class" href = getInput "href" limit, depth :: CGI (Maybe Int) limit = readInput "limit" depth = readInput "depth" from :: CGI (Maybe PGF.Language) from = getLang "from" to :: CGI [PGF.Language] to = getLangs "to" trie :: CGI Bool trie = maybe False toBool # getInput "trie" getLangs :: String -> CGI [PGF.Language] getLangs i = mapM readLang . maybe [] words =<< getInput i getLang :: String -> CGI (Maybe PGF.Language) getLang i = do mlang <- getInput i case mlang of Just l@(_:_) -> Just # readLang l _ -> return Nothing readLang :: String -> CGI PGF.Language readLang l = case PGF.readLanguage l of Nothing -> throwCGIError 400 "Bad language" ["Bad language: " ++ l] Just lang | lang `elem` PGF.languages pgf -> return lang | otherwise -> throwCGIError 400 "Unknown language" ["Unknown language: " ++ l] getIncludePrintNames :: CGI Bool getIncludePrintNames = maybe False (const True) # getInput "printnames" graphvizOptions = PGF.GraphvizOptions # bool "noleaves" % bool "nofun" % bool "nocat" % string "nodefont" % string "leaffont" % string "nodecolor" % string "leafcolor" % string "nodeedgestyle" % string "leafedgestyle" where string name = maybe "" id # getInput name bool name = maybe False toBool # getInput name toBool s = s `elem` ["","yes","true","True"] errorMissingId = throwCGIError 400 "Missing identifier" [] format def = maybe def id # getInput "format" -- Hook for simple extensions of the PGF service doExternal Nothing input = throwCGIError 400 "Unknown external command" ["Unknown external command"] doExternal (Just cmd) input = do liftIO $ logError ("External command: "++cmd) cmds <- liftIO $ (fmap lines $ readFile "external_services") `catchIOE` const (return []) liftIO $ logError ("External services: "++show cmds) if cmd `elem` cmds then ok else err where err = throwCGIError 400 "Unknown external command" ["Unknown external command: "++cmd] ok = do let tmpfile1 = "external_input.txt" tmpfile2 = "external_output.txt" liftIO $ writeFile "external_input.txt" input liftIO $ system $ cmd ++ " " ++ tmpfile1 ++ " > " ++ tmpfile2 liftIO $ removeFile tmpfile1 r <- outputJSONP =<< liftIO (readFile tmpfile2) liftIO $ removeFile tmpfile2 return r doTranslate :: PGF -> String -> Maybe PGF.Type -> Maybe PGF.Language -> [PGF.Language] -> Maybe Int -> Bool -> JSValue doTranslate pgf input mcat mfrom tos mlimit trie = showJSON [makeObj ("from".=from : "brackets".=bs : jsonTranslateOutput po) | (from,po,bs) <- parse' pgf input mcat mfrom] where jsonTranslateOutput output = case output of PGF.ParseOk trees -> addTrie trie trees++ ["translations".= [makeObj ["tree".=tree, "linearizations".= [makeObj ["to".=to, "text".=text, "brackets".=bs] | (to,text,bs)<- linearizeAndBind pgf tos tree]] | tree <- maybe id take mlimit trees]] PGF.ParseIncomplete -> ["incomplete".=True] PGF.ParseFailed n -> ["parseFailed".=n] PGF.TypeError errs -> jsonTypeErrors errs jsonTypeErrors errs = ["typeErrors".= [makeObj ["fid".=fid, "msg".=show (PGF.ppTcError err)] | (fid,err) <- errs]] -- used in phrasebook doTranslateGroup :: PGF -> String -> Maybe PGF.Type -> Maybe PGF.Language -> [PGF.Language] -> Maybe Int -> JSValue doTranslateGroup pgf input mcat mfrom tos mlimit = showJSON [makeObj ["from".=langOnly (PGF.showLanguage from), "to".=langOnly (PGF.showLanguage to), "linearizations".= [toJSObject (("text", doText alt) : disamb lg from ts) | (ts,alt) <- output, let lg = length output] ] | (from,po,bs) <- parse' pgf input mcat mfrom, (to,output) <- groupResults [(t, linearizeAndBind pgf tos t) | t <- case po of {PGF.ParseOk ts -> maybe id take mlimit ts; _ -> []}] ] where groupResults = Map.toList . foldr more Map.empty . start . collect where collect tls = [(t,(l,s)) | (t,ls) <- tls, (l,s,_) <- ls, notDisamb l] start ls = [(l,[([t],s)]) | (t,(l,s)) <- ls] more (l,s) = Map.insertWith (\ [([t],x)] xs -> insertAlt t x xs) l s insertAlt t x xs = case xs of (ts,y):xs2 -> if x==y then (t:ts,y):xs2 -- if string is there add only tree else (ts,y) : insertAlt t x xs2 _ -> [([t],x)] doText s = case s of c:cs | elem (last s) ".?!" -> toUpper c : init (init cs) ++ [last s] _ -> s langOnly = reverse . take 3 . reverse disamb lg from ts = if lg < 2 then [] else [("tree", "-- " ++ groupDisambs [doText (disambLang from t) | t <- ts])] groupDisambs = unwords . intersperse "/" disambLang f t = let disfl lang = PGF.mkCId ("Disamb" ++ lang) disf = disfl (PGF.showLanguage f) disfEng = disfl (reverse (drop 3 (reverse (PGF.showLanguage f))) ++ "Eng") in if elem disf (PGF.languages pgf) -- if Disamb f exists use it then PGF.linearize pgf disf t else if elem disfEng (PGF.languages pgf) -- else try DisambEng then PGF.linearize pgf disfEng t else "AST " ++ PGF.showExpr [] t -- else show abstract tree notDisamb = (/="Disamb") . take 6 . PGF.showLanguage doParse :: PGF -> String -> Maybe PGF.Type -> Maybe PGF.Language -> Maybe Int -> Bool -> JSValue doParse pgf input mcat mfrom mlimit trie = showJSON $ map makeObj ["from".=from : "brackets".=bs : jsonParseOutput po | (from,po,bs) <- parse' pgf input mcat mfrom] where jsonParseOutput output = case output of PGF.ParseOk trees -> ["trees".=maybe id take mlimit trees] ++addTrie trie trees PGF.TypeError errs -> jsonTypeErrors errs PGF.ParseIncomplete -> ["incomlete".=True] PGF.ParseFailed n -> ["parseFailed".=n] addTrie trie trees = ["trie".=map head (PGF.toTrie (map PGF.toATree trees))|trie] doComplete :: PGF -> String -> Maybe PGF.Type -> Maybe PGF.Language -> Maybe Int -> JSValue doComplete pgf input mcat mfrom mlimit = showJSON [makeObj ["from".=from, "brackets".=bs, "completions".=cs, "text".=s] | from <- froms, let (bs,s,cs) = complete' pgf from cat mlimit input] where froms = maybe (PGF.languages pgf) (:[]) mfrom cat = fromMaybe (PGF.startCat pgf) mcat doLinearize :: PGF -> PGF.Tree -> [PGF.Language] -> JSValue doLinearize pgf tree tos = showJSON [makeObj ["to".=to, "text".=text,"brackets".=bs] | (to,text,bs) <- linearizeAndBind pgf tos tree] doLinearizes :: PGF -> PGF.Tree -> [PGF.Language] -> JSValue doLinearizes pgf tree tos = showJSON [makeObj ["to".=to, "texts".=map doBind texts] | (to,texts) <- linearizes' pgf tos tree] where linearizes' pgf tos tree = [(to,lins to (transfer to tree)) | to <- langs] where langs = if null tos then PGF.languages pgf else tos lins to = nub . concatMap (map snd) . PGF.tabularLinearizes pgf to doLinearizeTabular :: PGF -> PGF.Tree -> [PGF.Language] -> JSValue doLinearizeTabular pgf tree tos = showJSON [makeObj ["to".=to, "table".=[makeObj ["params".=ps,"texts".=ts] | (ps,ts)<-texts]] | (to,texts) <- linearizeTabular pgf tos tree] doRandom :: PGF -> Maybe PGF.Type -> Maybe Int -> Maybe Int -> [PGF.Language] -> IO JSValue doRandom pgf mcat mdepth mlimit tos = do g <- newStdGen let trees = PGF.generateRandomDepth g pgf cat (Just depth) return $ showJSON [makeObj ["tree".=PGF.showExpr [] tree, "linearizations".= doLinearizes pgf tree tos] | tree <- limit trees] where cat = fromMaybe (PGF.startCat pgf) mcat limit = take (fromMaybe 1 mlimit) depth = fromMaybe 4 mdepth doGenerate :: PGF -> Maybe PGF.Type -> Maybe Int -> Maybe Int -> [PGF.Language] -> JSValue doGenerate pgf mcat mdepth mlimit tos = showJSON [makeObj ["tree".=PGF.showExpr [] tree, "linearizations".= [makeObj ["to".=to, "text".=text] | (to,text,bs) <- linearizeAndBind pgf tos tree]] | tree <- limit trees] where trees = PGF.generateAllDepth pgf cat (Just depth) cat = fromMaybe (PGF.startCat pgf) mcat limit = take (fromMaybe 1 mlimit) depth = fromMaybe 4 mdepth doGrammar :: PGF -> Maybe (Accept Language) -> JSValue doGrammar pgf macc = showJSON $ makeObj ["name".=PGF.abstractName pgf, "userLanguage".=selectLanguage pgf macc, "startcat".=PGF.showType [] (PGF.startCat pgf), "categories".=categories, "functions".=functions, "languages".=languages] where languages = [makeObj ["name".= l, "languageCode".= fromMaybe "" (PGF.languageCode pgf l)] | l <- PGF.languages pgf] categories = [PGF.showCId cat | cat <- PGF.categories pgf] functions = [PGF.showCId fun | fun <- PGF.functions pgf] outputGraphviz code = do fmt <- format "png" case fmt of "gv" -> outputPlain code _ -> outputFPS' fmt =<< liftIO (pipeIt2graphviz fmt code) where outputFPS' fmt bs = do setHeader "Content-Type" (mimeType fmt) outputFPS bs mimeType fmt = case fmt of "png" -> "image/png" "gif" -> "image/gif" "svg" -> "image/svg+xml" -- ... _ -> "application/binary" abstrTree pgf opts tree = PGF.graphvizAbstractTree pgf opts' tree where opts' = (not (PGF.noFun opts),not (PGF.noCat opts)) parseTree pgf lang opts tree = PGF.graphvizParseTree pgf lang opts tree alignment pgf tree tos = PGF.graphvizAlignment pgf tos' tree where tos' = if null tos then PGF.languages pgf else tos pipeIt2graphviz :: String -> String -> IO BS.ByteString pipeIt2graphviz fmt code = do (Just inh, Just outh, _, pid) <- createProcess (proc "dot" ["-T",fmt]) { std_in = CreatePipe, std_out = CreatePipe, std_err = Inherit } hSetBinaryMode outh True hSetEncoding inh utf8 -- fork off a thread to start consuming the output output <- BS.hGetContents outh outMVar <- newEmptyMVar _ <- forkIO $ E.evaluate (BS.length output) >> putMVar outMVar () -- now write and flush any input hPutStr inh code hFlush inh hClose inh -- done with stdin -- wait on the output takeMVar outMVar hClose outh -- wait on the process ex <- waitForProcess pid case ex of ExitSuccess -> return output ExitFailure r -> fail ("pipeIt2graphviz: (exit " ++ show r ++ ")") browse1json pgf id pn = makeObj . maybe [] obj $ PGF.browse pgf id where obj (def,ps,cs) = if pn then (baseobj ++ pnames) else baseobj where baseobj = ["def".=def, "producers".=ps, "consumers".=cs] pnames = ["printnames".=makeObj [(show lang).=PGF.showPrintName pgf lang id | lang <- PGF.languages pgf]] doBrowse pgf (Just id) _ _ "json" pn = outputJSONP $ browse1json pgf id pn doBrowse pgf Nothing _ _ "json" pn = outputJSONP $ makeObj ["cats".=all (PGF.categories pgf), "funs".=all (PGF.functions pgf)] where all = makeObj . map one one id = PGF.showCId id.=browse1json pgf id pn doBrowse pgf Nothing cssClass href _ pn = errorMissingId doBrowse pgf (Just id) cssClass href _ pn = -- default to "html" format outputHTML $ case PGF.browse pgf id of Just (def,ps,cs) -> "
"++annotate def++"\n"++ syntax++ (if not (null ps) then "
"++annotateCIds ps++"
\n" else "")++ (if not (null cs) then ""++annotateCIds cs++"
\n" else "")++ (if pn then ""++annotatePrintNames++"
\n" else "") Nothing -> "" where syntax = case PGF.functionType pgf id of Just ty -> let (hypos,_,_) = PGF.unType ty e = PGF.mkApp id (snd $ mapAccumL mkArg (1,1) hypos) rows = ["| "++PGF.showCId (PGF.abstractName pgf)++" | "++ ""++PGF.showExpr [] e++" | "++ "