1
0
forked from GitHub/gf-rgl

the script for checking Icelandic verbs wrt Wiktionary

This commit is contained in:
Aarne Ranta
2019-08-06 17:07:00 +02:00
parent 8b84b89941
commit 7bdc01f45a

View File

@@ -0,0 +1,137 @@
import qualified Data.Map
import qualified Data.Char
import qualified Data.List
-- AR 2019-08-06
-- checking IrregIce wrt Wikipedia:
--
-- > getAllWikt, producing wiktVerbDataIce.txt
-- > correctAllRGL, producing rgl-corrected.tmp, rgl-corrected.gf
-- first do:
-- (gf alltenses/IrregIce.gfo) > gt -cat=V | l -table | wf -file="irregsIce.tmp"
-- % wget https://en.wiktionary.org/wiki/<each-verb-infinitive> from the 'verbs' list given below
main = correctAllRGL
correctAllRGL = do
ls <- readFile "irregsIce.tmp" >>= return . lines
let vs = getGroups ls
let vis = [(actInf v,v) | v <- vs]
cvs <- mapM (uncurry rgl2wik) vis
writeFile "rgl-corrected.tmp" $ unlines $ concat cvs
let cgf = map (uncurry fixedRGL) (zip (map fst vis) cvs)
writeFile "rgl-corrected.gf" $ concat $ Data.List.intersperse ", " [f++"_V" | (Just f,_) <- cgf]
appendFile "rgl-corrected.gf" $ concat $ map (unlines . snd) $ cgf
-- this produces an inspectable file with extracted Wiktionary data, including deviating layouts
getAllWikt = do
vs <- mapM getWikt verbs
writeFile "wiktVerbDataIce.txt" $ unlines $ concat $ Data.List.intersperse ["------"] vs
verbs = ["aka","ala","auka","ausa","bera","binda","bi\240ja","bjarga","bl\225sa","brenna","bresta","brj\243ta","b\237ta","b\237\240a","b\250a","detta","deyja","draga","drekka","drepa","drj\250pa","dr\237fa","eiga","falla","fara","fela","finna","fj\250ka","flj\243ta","flj\250ga","fl\225","frj\243sa","f\225","gala","ganga","gefa","geta","geyja","gjalda","gjalla","gj\243sa","grafa","gr\225ta","gr\237pa","gr\243a","halda","hanga","hefja","heita","hlaupa","hla\240a","hlj\243ta","hl\230ja","hnj\243ta","hn\237ga","hrinda","hr\237fa","hr\237na","hr\246kkva","hverfa","hv\237na","h\246ggva","kala","kefja","kj\243sa","klj\250fa","koma","kunna","kve\240a","kv\237\240a","leika","leka","lesa","liggja","lj\250ga","lj\250ka","l\225ta","l\237ta","l\237\240a","l\250ka","l\250ta","mala","mega","meta","muna","munu","m\237ga","nema","nj\243ta","n\250a","reka","renna","rj\250fa","r\225\240a","r\237sa","r\237\240a","r\243a","sitja","sj\225","sj\243\240a","sj\250ga","skafa","skaka","skapa","skella","skera","skj\225lfa","skreppa","skr\237\240a","skulu","sk\237na","sleppa","sl\225","sl\237ta","sl\246kkva","smella","smj\250ga","snerta","sn\237\240a","sn\250a","sofa","spinna","spretta","springa","sp\250a","standa","stela","stinga","strj\250ka","st\237ga","st\246kkva","svelta","sverfa","sverja","sv\237fa","sv\237kja","syngja","s\237ga","s\246kkva","s\250pa","taka","tro\240a","unna","vaxa","vega","vella","velta","vera","verpa","ver\240a","vilja","vinda","vinna","vita","v\237kja","\233ta","\254iggja","\254j\243ta","\254rj\243ta","\254r\237fa","\254urfa","\254verra","\254vo"]
-- not all of these are in Wiktionary
rgl_verbs = ["aka","ala","auka","ausa","bera","binda","bi\240ja","bjarga","bl\225sa","brenna","bresta","brj\243ta","b\237ta","b\237\240a","b\250a","detta","deyja","draga","drekka","drepa","drj\250pa","dr\237fa","eiga","falla","fara","fela","finna","fj\250ka","flj\243ta","flj\250ga","fl\225","frj\243sa","f\225","gala","ganga","gefa","geta","geyja","gjalda","gjalla","gj\243sa","grafa","gr\225ta","gr\237pa","gr\243a","halda","hanga","hefja","heita","hlaupa","hla\240a","hlj\243ta","hl\230ja","hnj\243ta","hn\237ga","hrinda","hr\237fa","hr\237na","hr\246kkva","hverfa","hv\237na","h\246ggva","kala","kefja","kj\243sa","klj\250fa","kl\230ja","koma","kunna","kve\240a","kv\237\240a","leika","leka","lesa","liggja","lj\250ga","lj\250ka","l\225ta","l\237ta","l\237\240a","l\250ka","l\250ta","mala","mega","meta","muna","munu","m\237ga","nema","nj\243ta","n\250a","reka","renna","rj\250fa","r\225\240a","r\237sa","r\237\240a","r\243a","sitja","sj\225","sj\243\240a","sj\250ga","skafa","skaka","skapa","skekja","skella","skera","skj\225lfa","skreppa","skr\237\240a","skulu","sk\237na","sleppa","sl\225","sl\237ta","sl\246kkva","smella","smj\250ga","snerta","sn\237\240a","sn\250a","sofa","spinna","spretta","springa","sp\250a","standa","stela","stinga","strj\250ka","st\237ga","st\246kkva","svelta","sverfa","sverja","sv\237fa","sv\237kja","syngja","s\237ga","s\246kkva","s\250pa","taka","tro\240a","unna","vaxa","vega","vella","velta","vera","verpa","ver\240a","vilja","vinda","vinna","vita","v\237kja","\233ta","\254iggja","\254j\243ta","\254rj\243ta","\254r\237fa","\254urfa","\254verra","\254vo"]
-- Wiktionary table for one verb
getWikt :: FilePath -> IO [String]
getWikt file =
if elem file verbs
then putStrLn ("YES: " ++ file) >> readFile file >>= checkPrefixes file . take 75 . dropWhile (not . isStart file) . map untag . getTD . lines
else putStrLn ("NO: " ++ file) >> return []
-- check that a Wiktionary file has expected layout
--- this is only a partial check
checkPrefixes :: String -> [String] -> IO [String]
checkPrefixes file ls =
if length ls < 75
then putStrLn ("too short " ++ file) >> return []
else case unexpectedWikLines ls of
[] -> return ls
us -> putStrLn (unlines (("unexpected layout in " ++ file) : us)) >> return []
-- remove HTML tags
untag s = case s of {'<':cs -> intag cs ; c:cs -> c:untag cs ; _ -> s}
where intag s = case s of '>':cs -> untag cs ; _:cs -> intag cs
-- consider td cells
getTD = filter (\l -> take 3 l == "<td")
-- start from the Icelandic infinitive form
isStart inf s = words s == ["",inf]
-- find the word from each td cell; return ZZ if not found
wform wik = last $ "ZZ":(filter (all Data.Char.isAlpha) wik)
-- find the Wiktionary forms for each branch in the RGL table
wforms ws = [(i, (if length ws < j then "XX" else wform (words (ws !! (j-1))))) | (i,j) <- rgl2wiktLines]
-- to correct one RGL table by reference to the Wiktionary data file
rgl2wik :: String -> [String] -> IO [String]
rgl2wik i v = do
fs <- getWikt i
let wmap = Data.Map.fromList (wforms fs)
putStrLn i
print $ Data.Map.size wmap
let facit = [unwords (gws ++ correctForm wmap (j,last gws)) | (j,gw) <- zip [1..] v, let gws = words gw]
return $ status i facit : facit
-- fix one entry, provided it has more than 0 deviations
fixedRGL :: String -> [String] -> (Maybe String,[String])
fixedRGL i (s:rs) =
if last (words s) == "0"
then (Nothing,[])
else (Just i, [
"lin " ++ i ++ "_V = OI." ++ i ++ "_V ** {",
" s = table {"
] ++
fixes rs ++
[
" vf => OI." ++ i ++ "_V.s ! vf",
" } ;",
" } ;"
]
)
-- produces the branch for a form that is fixed
fixes rs = [" " ++ vf ++ " => \"" ++ form ++ "\" ; --- " ++ reverse wr !! 2 |
r <- rs, let wr = words r, head wr == "s", elem "*" wr, notElem "XX" wr, notElem "ZZ" wr,
let vf = takeWhile (/=')') $ drop 1 $ dropWhile (/='(') r,
let form = last wr
]
-- line in RGL table -> line in Wik data
rgl2wiktLines = [(1,1),(2,38),(3,6),(4,10),(5,14),(6,7),(7,11),(8,15),(9,8),(10,12),(11,16),(12,9),(13,13),(14,17),(15,43),(16,47),(17,51),(18,44),(19,48),(20,52),(21,45),(22,49),(23,53),(24,46),(25,50),(26,54),(27,19),(28,23),(29,27),(30,20),(31,24),(32,28),(33,21),(34,25),(35,29),(36,22),(37,26),(38,30),(39,56),(40,60),(41,64),(42,57),(43,61),(44,65),(45,58),(46,62),(47,66),(48,59),(49,63),(50,67),(51,35),(52,33),(53,72),(54,70),(55,2),(56,39),(81,75),(105,3)]
-- prefixes on each line of valid Wik data, as extracted from the verb "fara"
expectedPrefixes = map unwords [["a\240"],[],[],[],[],["\233g"],["vi\240"],["\233g"],["vi\240"],["\254\250"],["\254i\240"],["\254\250"],["\254i\240"],["hann"],["\254eir"],["hann"],["\254eir"],[],["\233g"],["vi\240"],["\233g"],["vi\240"],["\254\250"],["\254i\240"],["\254\250"],["\254i\240"],["hann"],["\254eir"],["hann"],["\254eir"],[],[],[],[],[],[],[],["a\240"],[],[],[],[],["\233g"],["vi\240"],["\233g"],["vi\240"],["\254\250"],["\254i\240"],["\254\250"],["\254i\240"],["hann"],["\254eir"],["hann"],["\254eir"],[],["\233g"],["vi\240"],["\233g"],["vi\240"],["\254\250"],["\254i\240"],["\254\250"],["\254i\240"],["hann"],["\254eir"],["hann"],["\254eir"],[],[],[],[],[],[],[],[],[],[],[],[],[],[],[],[],[],[],[],[],[],[],[],[],[],[],[],[],[],[],[],[],[],[],[],[],[],[],[],[],[],[],[],[],[],[],[],[],[],[],[],[],[],[],[]]
-- mark Wiktionary data lines not matching the layout, and therefore ignored in the main function
unexpectedWikLines ws = [l ++ " --" ++ p | (l,p) <- zip ws expectedPrefixes, not (Data.List.isPrefixOf p l)]
-- if a form is not found in Wikt, return ?? ; this is not used as a correction
correctForm wmap (j,w) = case Data.Map.lookup j wmap of
Just v | v == w -> []
Just v -> ["*",v]
_ -> ["??"]
--- used on the manually marked Wiktionary list for the verb "fara"; this became the expected layout
wformsIfMarked ws = [(read (last ww) :: Int, (wform ww, i)) |
(i,w) <- zip [1..] ws, let ww = words w, not (null ww), all Data.Char.isDigit (last ww)]
-- the status is the number of deviations in a verb's table
status :: String -> [String] -> String
status inf ls = unwords $ "STATUS" : inf : [show (deviations ls)]
-- deviations in the intermediate file are marked with *; XX and ZZ are not included as correct forms
deviations :: [String] -> Int
deviations ls = length [l | l <- ls, let r = reverse (words l), r!!1 == "*", notElem (head r) ["XX","ZZ"]]
-- an RGL verb table has length 105
getGroups ls = let (v,vs) = splitAt 105 ls in if null v then [] else v:getGroups vs
actInf v@(i:_) = last (words i)