attoparsec:"嵌套"解析器 - 使用不同的解析器解析输入的子集

Emm*_*ery 7 haskell attoparsec

事实上,我很确定我使用的是错误的术语.这是我想要解决的问题:markdown格式的解析器,以及它的子集.

我的问题是blockquote功能.blockquote中的每一行都以>; 否则一切都是降价文件中的正常结构.

您不能单独查看单独的行,因为您需要将段落与法线分开,例如

> a
> b
Run Code Online (Sandbox Code Playgroud)

是不一样的

> a
>
> b
Run Code Online (Sandbox Code Playgroud)

和类似的东西(如果列表是块引用,你不需要x列表,但是一个列表包含x个元素).一种自然而琐碎的方法是"取消" >符号,自己解析块引用,忽略它周围的任何东西,用BlockQuote类型构造函数包装它,将它放在外部AST中并继续解析原始输入.pango如果我没错,那就是这样:

https://hackage.haskell.org/package/pandoc-1.14.0.4/docs/src/Text-Pandoc-Readers-Markdown.html#blockQuote

blockQuote :: MarkdownParser (F Blocks)
blockQuote = do
  raw <- emailBlockQuote
  -- parse the extracted block, which may contain various block elements:
  contents <- parseFromString parseBlocks $ (intercalate "\n" raw) ++ "\n\n"
  return $ B.blockQuote <$> contents
Run Code Online (Sandbox Code Playgroud)

然后:

http://hackage.haskell.org/package/pandoc-1.5.1/docs/src/Text-Pandoc-Shared.html#parseFromString

-- | Parse contents of 'str' using 'parser' and return result.
parseFromString :: GenParser tok st a -> [tok] -> GenParser tok st a
parseFromString parser str = do
  oldPos <- getPosition
  oldInput <- getInput
  setInput str
  result <- parser
  setInput oldInput
  setPosition oldPos
  return result
Run Code Online (Sandbox Code Playgroud)

现在parseFromString看起来相当哈克我再说,这也是Parsec没有attoparsec,所以我不能在我的项目中使用它.我不确定如何Text从blockquote中获取它并解析它并返回解析结果,以便它"适合"当前解析.似乎不可能?

我一直在使用Google的问题,我认为pipes-parseconduit可以在该地区帮助,虽然我努力寻找例子,我所看到的出现大大减少好看比"纯"秒差距/ attoparsec解析器.

解析blockquotes的其他选项是重写通常的解析器但是使用>catch ...复杂和重复很多.解析blockquotes分别计算每一行并编写一些杂乱的"合并"函数.或者解析为第一个包含blockquotes的AST,作为Text第一个BlockquoteText类型构造函数内部,等待转换,它们将被单独解析,不是很优雅,但它具有简单的优点,这确实可以算作一些东西.

我可能会选择后者,但肯定有更好的方法吗?

tre*_*ook 4

我也问过自己同样的问题。为什么没有像您所描述的那样的嵌套解析器的标准组合器?我的默认模式是信任包作者,特别是当该作者还与他人共同编写了“Real World Haskell”时。如果缺少这样一个明显的功能,也许是设计使然,我应该寻找更好的方法。然而,我已经成功地说服自己,这样一个方便的组合器基本上是无害的。当全有或全无类型解析器适合内部解析时很有用。

执行

import Data.Attoparsec.Text
import qualified Data.Text as T
import Data.Text(Text)
import Control.Applicative
Run Code Online (Sandbox Code Playgroud)

我已将所需的功能划分为两个解析器。第一个,constP对某些给定文本执行“就地”解析。它用(来自替代方案)替换常量解析器的失败empty,但没有其他副作用。

constP :: Parser a -> Text -> Parser a
constP p t = case parseOnly p t of
  Left _ -> empty
  Right a -> return a
Run Code Online (Sandbox Code Playgroud)

第二部分来自parseOf,它根据外部解析的结果执行恒定的内部解析。这里的替代方案empty允许返回失败的解析而不消耗任何输入。

parseOf :: Parser Text -> Parser a -> Parser a
parseOf ptxt pa = bothParse <|> empty
  where
    bothParse = ptxt >>= constP pa
Run Code Online (Sandbox Code Playgroud)

块引用降价可以按照所需的方式编写。此实现需要完全解析生成的块。

blockQuoteMarkdown :: Parser [[Double]]
blockQuoteMarkdown = parseOf blockQuote ( markdownSurrogate <* 
                                          endOfInput
                                        )
Run Code Online (Sandbox Code Playgroud)

我只是实现了一个空格分隔双精度数的快速解析器,而不是实际的降价解析器。解析器的复杂性来自于允许最后一个非空行,无论是否以新行结束。

markdownSurrogate :: Parser [[Double]]
markdownSurrogate = do
  lns <- many (mdLine <* endOfLine)
  option lns ((lns ++) . pure <$> mdLine1)
  where
    mdLine = sepBy double (satisfy (==' '))
    mdLine1 = sepBy1 double (satisfy (==' '))
Run Code Online (Sandbox Code Playgroud)

这两个解析器负责返回块引用内部的文本。

blockQuote :: Parser Text
blockQuote = T.unlines <$> many blockLine

blockLine :: Parser Text
blockLine = char '>' *> takeTill isEndOfLine <* endOfLine
Run Code Online (Sandbox Code Playgroud)

最后,对解析器进行测试。

parseMain :: IO ()
parseMain = do

  putStrLn ""
  doParse "a" markdownSurrogate a
  doParse "_" markdownSurrogate ""
  doParse "b" markdownSurrogate b
  doParse "ab" markdownSurrogate ab
  doParse "a_b" markdownSurrogate a_b
  doParse "badMarkdown x" markdownSurrogate x
  doParse "badMarkdown axb" markdownSurrogate axb

  putStrLn ""
  doParse "BlockQuote ab" blockQuoteMarkdown $ toBlockQuote ab
  doParse "BlockQuote a_b" blockQuoteMarkdown $ toBlockQuote a_b
  doParse "BlockQuote axb" blockQuoteMarkdown $ toBlockQuote axb
  where
    a = "7 3 1"
    b = "4 4 4"
    x = "a b c"

    ab = T.unlines [a,b]
    a_b = T.unlines [a,"",b]
    axb = T.unlines [a,x,b]

    doParse desc p str = do
      print $ T.concat ["Parsing ",desc,": \"",str,"\""]
      let i = parse (p <* endOfInput ) str
      print $ feed i ""

    toBlockQuote = T.unlines
                 . map (T.cons '>')
                 . T.lines

*Main> parseMain

"Parsing a: \"7 3 1\""
Done "" [[7.0,3.0,1.0]]
"Parsing _: \"\""
Done "" []
"Parsing b: \"4 4 4\""
Done "" [[4.0,4.0,4.0]]
"Parsing ab: \"7 3 1\n4 4 4\n\""
Done "" [[7.0,3.0,1.0],[4.0,4.0,4.0]]
"Parsing a_b: \"7 3 1\n\n4 4 4\n\""
Done "" [[7.0,3.0,1.0],[],[4.0,4.0,4.0]]
"Parsing badMarkdown x: \"a b c\""
Fail "a b c" [] "endOfInput"
"Parsing badMarkdown axb: \"7 3 1\na b c\n4 4 4\n\""
Fail "a b c\n4 4 4\n" [] "endOfInput"

"Parsing BlockQuote ab: \">7 3 1\n>4 4 4\n\""
Done "" [[7.0,3.0,1.0],[4.0,4.0,4.0]]
"Parsing BlockQuote a_b: \">7 3 1\n>\n>4 4 4\n\""
Done "" [[7.0,3.0,1.0],[],[4.0,4.0,4.0]]
"Parsing BlockQuote axb: \">7 3 1\n>a b c\n>4 4 4\n\""
Fail ">7 3 1\n>a b c\n>4 4 4\n" [] "Failed reading: empty"
Run Code Online (Sandbox Code Playgroud)

讨论

显着的差异在于失败的语义。例如,解析axb和 blockquoted时axb,分别是以下两个字符串

7 3 1
a b c
4 4 4
Run Code Online (Sandbox Code Playgroud)

> 7 3 1
> a b c
> 4 4 4
Run Code Online (Sandbox Code Playgroud)

Markdown 解析结果为

Fail "a b c\n4 4 4\n" [] "endOfInput"
Run Code Online (Sandbox Code Playgroud)

而引用的结果是

Fail ">7 3 1\n>a b c\n>4 4 4\n" [] "Failed reading: empty"
Run Code Online (Sandbox Code Playgroud)

降价消耗了“7 3 1\n”,但是在引用的失败中没有报告这一点。相反,失败就变得要么全有,要么一无所有。

同样,在部分成功的情况下,不允许处理未解析的文本。但考虑到用例,我认为没有必要这样做。例如,如果解析类似于以下内容

"{ <tok> unhandled }more to parse"
Run Code Online (Sandbox Code Playgroud)

其中{}表示识别的块引用上下文,并<tok>在该内部上下文中进行解析。要想取得部分成功,就必须将“未处理”从块引用上下文中剔除,并以某种方式将其与“更多要解析的内容”结合起来。

我认为没有通用的方法可以做到这一点,但可以通过选择内部解析器返回类型来实现。例如,通过某些解析器parseOf blockP innP :: Parser (<tok>,Maybe Text)。但是,如果出现这种需要,我希望有比嵌套解析器更好的方法来处理这种情况。

还可能有人担心 attoparsec 部分解析的丢失。也就是说,constPuses的实现parseOnly会将解析返回Fail折叠Partial到单个Left失败状态。换句话说,我们失去了向内部解析器提供更多可用文本的能力。但是,请注意,要解析的文本本身就是外部解析的结果;只有在将足够的文本输入到外部解析后,它才可用。所以这也不应该是一个问题。