Completely transitioned to LALRPOP and new declare macro.
This commit is contained in:
parent
cb7e148310
commit
7d1e95f262
|
@ -8,8 +8,8 @@ build = "build.rs"
|
||||||
proc-macro = true
|
proc-macro = true
|
||||||
|
|
||||||
[dependencies]
|
[dependencies]
|
||||||
pom = "2.0.1"
|
|
||||||
lalrpop-util = "0.16.1"
|
lalrpop-util = "0.16.1"
|
||||||
|
ansi_term = "0.11.0"
|
||||||
|
|
||||||
[build-dependencies]
|
[build-dependencies]
|
||||||
lalrpop = "0.16.1"
|
lalrpop = "0.16.1"
|
||||||
|
|
|
@ -1,11 +1,10 @@
|
||||||
use pom::combinator::*;
|
use proc_macro::{quote, Ident, Literal, TokenStream, TokenTree};
|
||||||
use pom::Parser;
|
|
||||||
use proc_macro::{quote, Group, Ident, Literal, TokenStream, TokenTree};
|
|
||||||
|
|
||||||
use config::global_attrs;
|
use config::global_attrs;
|
||||||
use lexer::{Lexer, ParseError, Token};
|
use error::ParseError;
|
||||||
|
use lexer::{Lexer, Token};
|
||||||
use map::StringyMap;
|
use map::StringyMap;
|
||||||
use parser::{self, *};
|
use parser;
|
||||||
|
|
||||||
// State
|
// State
|
||||||
|
|
||||||
|
@ -327,60 +326,6 @@ impl Declare {
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
// Parser
|
pub fn expand_declare(input: &[Token]) -> Result<Vec<Declare>, ParseError> {
|
||||||
|
|
||||||
fn declare_attrs<'a>() -> Combinator<impl Parser<'a, TokenTree, Output = Vec<(Ident, TokenStream)>>>
|
|
||||||
{
|
|
||||||
group().map(|group: Group| -> Vec<(Ident, TokenStream)> {
|
|
||||||
let attr = ident() - punct(':') + type_spec() - punct(',').opt();
|
|
||||||
let parser = attr.repeat(0..);
|
|
||||||
let input: Vec<TokenTree> = group.stream().into_iter().collect();
|
|
||||||
// FIXME the borrow checker won't let me use plain &input, it seems like a bug.
|
|
||||||
// It works in Rust 2018, so please get rid of this unsafe block when it stabilises.
|
|
||||||
parser
|
|
||||||
.parse(unsafe { &*(input.as_slice() as *const _) })
|
|
||||||
.unwrap()
|
|
||||||
})
|
|
||||||
}
|
|
||||||
|
|
||||||
fn declare_children<'a>() -> Combinator<impl Parser<'a, TokenTree, Output = Vec<Ident>>> {
|
|
||||||
group().map(|group: Group| {
|
|
||||||
let input: Vec<TokenTree> = group.stream().into_iter().collect();
|
|
||||||
let children = (ident() - punct(',').opt()).repeat(0..);
|
|
||||||
let result = children.parse(&input);
|
|
||||||
result.unwrap()
|
|
||||||
})
|
|
||||||
}
|
|
||||||
|
|
||||||
fn declare_traits<'a>() -> Combinator<impl Parser<'a, TokenTree, Output = Vec<TokenStream>>> {
|
|
||||||
group().map(|group: Group| {
|
|
||||||
let input: Vec<TokenTree> = group.stream().into_iter().collect();
|
|
||||||
let traits = (type_spec() - punct(',').opt()).repeat(0..);
|
|
||||||
let result = traits.parse(&input);
|
|
||||||
result.unwrap()
|
|
||||||
})
|
|
||||||
}
|
|
||||||
|
|
||||||
fn declare<'a>() -> Combinator<impl Parser<'a, TokenTree, Output = Declare>> {
|
|
||||||
(ident() + declare_attrs() + declare_children() + declare_traits().opt() + type_spec().opt())
|
|
||||||
.map(|((((name, attrs), children), traits), child_type)| {
|
|
||||||
let mut declare = Declare::new(name);
|
|
||||||
for (key, value) in attrs {
|
|
||||||
declare.attrs.insert(key, value);
|
|
||||||
}
|
|
||||||
for child in children {
|
|
||||||
declare.req_children.push(child);
|
|
||||||
}
|
|
||||||
declare.opt_children = child_type;
|
|
||||||
declare.traits = traits.unwrap_or_default();
|
|
||||||
declare
|
|
||||||
})
|
|
||||||
}
|
|
||||||
|
|
||||||
pub fn expand_declare(input: &[TokenTree]) -> pom::Result<TokenStream> {
|
|
||||||
declare().parse(input).map(|decl| decl.into_token_stream())
|
|
||||||
}
|
|
||||||
|
|
||||||
pub fn expand_declare_lalrpop(input: &[Token]) -> Result<Vec<Declare>, ParseError> {
|
|
||||||
parser::grammar::DeclarationsParser::new().parse(Lexer::new(input))
|
parser::grammar::DeclarationsParser::new().parse(Lexer::new(input))
|
||||||
}
|
}
|
||||||
|
|
|
@ -0,0 +1,88 @@
|
||||||
|
use ansi_term::Style;
|
||||||
|
use lalrpop_util::ParseError::*;
|
||||||
|
use lexer::Token;
|
||||||
|
use proc_macro::{Diagnostic, Ident, Level};
|
||||||
|
|
||||||
|
pub type ParseError = lalrpop_util::ParseError<usize, Token, HtmlParseError>;
|
||||||
|
|
||||||
|
#[derive(Debug)]
|
||||||
|
pub enum HtmlParseError {
|
||||||
|
TagMismatch { open: Ident, close: Ident },
|
||||||
|
}
|
||||||
|
|
||||||
|
fn pprint_token(token: &str) -> &str {
|
||||||
|
match token {
|
||||||
|
"BraceGroupToken" => "code block",
|
||||||
|
"LiteralToken" => "literal",
|
||||||
|
"IdentToken" => "identifier",
|
||||||
|
a => a,
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
fn pprint_tokens(tokens: &[String]) -> String {
|
||||||
|
let tokens: Vec<&str> = tokens.iter().map(|s| pprint_token(&s)).collect();
|
||||||
|
if tokens.len() > 1 {
|
||||||
|
let start = tokens[..tokens.len() - 1].join(", ");
|
||||||
|
let end = &tokens[tokens.len() - 1];
|
||||||
|
format!("{} or {}", start, end)
|
||||||
|
} else {
|
||||||
|
tokens[0].to_string()
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
fn is_in_node_position(tokens: &[String]) -> bool {
|
||||||
|
use std::collections::HashSet;
|
||||||
|
let input: HashSet<&str> = tokens.iter().map(String::as_str).collect();
|
||||||
|
let output: HashSet<&str> = ["\"<\"", "BraceGroupToken", "LiteralToken"]
|
||||||
|
.iter()
|
||||||
|
.cloned()
|
||||||
|
.collect();
|
||||||
|
input == output
|
||||||
|
}
|
||||||
|
|
||||||
|
pub fn parse_error(input: &[Token], error: &ParseError) -> Diagnostic {
|
||||||
|
match error {
|
||||||
|
InvalidToken { location } => {
|
||||||
|
let loc = &input[*location];
|
||||||
|
Diagnostic::spanned(loc.span(), Level::Error, "invalid token")
|
||||||
|
}
|
||||||
|
UnrecognizedToken {
|
||||||
|
token: None,
|
||||||
|
expected,
|
||||||
|
} => {
|
||||||
|
let msg = format!("missing {}", pprint_tokens(&expected));
|
||||||
|
Diagnostic::spanned(
|
||||||
|
input[0].span().join(input[input.len() - 1].span()).unwrap(),
|
||||||
|
Level::Error,
|
||||||
|
"unexpected end of macro",
|
||||||
|
)
|
||||||
|
.help(msg)
|
||||||
|
}
|
||||||
|
UnrecognizedToken {
|
||||||
|
token: Some((_, token, _)),
|
||||||
|
expected,
|
||||||
|
} => {
|
||||||
|
let msg = format!("expected {}", pprint_tokens(&expected));
|
||||||
|
let mut diag = Diagnostic::spanned(token.span(), Level::Error, msg);
|
||||||
|
if is_in_node_position(expected) && token.is_ident() {
|
||||||
|
// special case: you probably meant to quote that text
|
||||||
|
diag = diag.help(format!(
|
||||||
|
"text nodes need to be quoted, eg. {}",
|
||||||
|
Style::new().bold().paint("<p>\"Hello Joe!\"</p>")
|
||||||
|
))
|
||||||
|
}
|
||||||
|
diag
|
||||||
|
}
|
||||||
|
ExtraToken {
|
||||||
|
token: (_, token, _),
|
||||||
|
} => Diagnostic::spanned(token.span(), Level::Error, "superfluous token"),
|
||||||
|
User {
|
||||||
|
error: HtmlParseError::TagMismatch { open, close },
|
||||||
|
} => Diagnostic::spanned(
|
||||||
|
close.span(),
|
||||||
|
Level::Error,
|
||||||
|
format!("expected closing tag '</{}>', found '</{}>'", open, close),
|
||||||
|
)
|
||||||
|
.span_help(open.span(), "opening tag is here:"),
|
||||||
|
}
|
||||||
|
}
|
|
@ -1,4 +1,5 @@
|
||||||
use lexer::{Token, to_stream, HtmlParseError, Keyword};
|
use lexer::{self, Token, to_stream};
|
||||||
|
use error::HtmlParseError;
|
||||||
use html::{Node, Element};
|
use html::{Node, Element};
|
||||||
use declare::Declare;
|
use declare::Declare;
|
||||||
use map::StringyMap;
|
use map::StringyMap;
|
||||||
|
@ -130,9 +131,9 @@ ParentTag: Element = <opening:OpeningTag> <children:Node*> <closing:ClosingTag>
|
||||||
children,
|
children,
|
||||||
})
|
})
|
||||||
} else {
|
} else {
|
||||||
Err(ParseError::User { error: HtmlParseError {
|
Err(ParseError::User { error: HtmlParseError::TagMismatch {
|
||||||
token: closing.into(),
|
open: name.into(),
|
||||||
message: format!("expected closing tag '</{}>', found '</{}>'", name.to_string(), closing_name),
|
close: closing.into(),
|
||||||
}})
|
}})
|
||||||
}
|
}
|
||||||
};
|
};
|
||||||
|
@ -260,7 +261,7 @@ extern {
|
||||||
type Location = usize;
|
type Location = usize;
|
||||||
type Error = HtmlParseError;
|
type Error = HtmlParseError;
|
||||||
|
|
||||||
enum Token {
|
enum lexer::Token {
|
||||||
"<" => Token::Punct('<', _),
|
"<" => Token::Punct('<', _),
|
||||||
">" => Token::Punct('>', _),
|
">" => Token::Punct('>', _),
|
||||||
"/" => Token::Punct('/', _),
|
"/" => Token::Punct('/', _),
|
||||||
|
@ -276,8 +277,8 @@ extern {
|
||||||
"}" => Token::GroupClose(Delimiter::Brace, _),
|
"}" => Token::GroupClose(Delimiter::Brace, _),
|
||||||
"[" => Token::GroupOpen(Delimiter::Bracket, _),
|
"[" => Token::GroupOpen(Delimiter::Bracket, _),
|
||||||
"]" => Token::GroupClose(Delimiter::Bracket, _),
|
"]" => Token::GroupClose(Delimiter::Bracket, _),
|
||||||
"in" => Token::Keyword(Keyword::In, _),
|
"in" => Token::Keyword(lexer::Keyword::In, _),
|
||||||
"with" => Token::Keyword(Keyword::With, _),
|
"with" => Token::Keyword(lexer::Keyword::With, _),
|
||||||
IdentToken => Token::Ident(_),
|
IdentToken => Token::Ident(_),
|
||||||
LiteralToken => Token::Literal(_),
|
LiteralToken => Token::Literal(_),
|
||||||
ParenGroupToken => Token::Group(Delimiter::Parenthesis, _),
|
ParenGroupToken => Token::Group(Delimiter::Parenthesis, _),
|
||||||
|
|
|
@ -3,7 +3,8 @@ use proc_macro::{
|
||||||
};
|
};
|
||||||
|
|
||||||
use config::required_children;
|
use config::required_children;
|
||||||
use lexer::{Lexer, ParseError, Token};
|
use error::ParseError;
|
||||||
|
use lexer::{Lexer, Token};
|
||||||
use map::StringyMap;
|
use map::StringyMap;
|
||||||
use parser::grammar;
|
use parser::grammar;
|
||||||
|
|
||||||
|
|
|
@ -1,10 +1,7 @@
|
||||||
use lalrpop_util::ParseError::*;
|
use error::HtmlParseError;
|
||||||
use proc_macro::{
|
use proc_macro::{Delimiter, Group, Ident, Literal, Punct, Span, TokenStream, TokenTree};
|
||||||
Delimiter, Diagnostic, Group, Ident, Level, Literal, Punct, Span, TokenStream, TokenTree,
|
|
||||||
};
|
|
||||||
|
|
||||||
pub type Spanned<Tok, Loc, Error> = Result<(Loc, Tok, Loc), Error>;
|
pub type Spanned<Tok, Loc, Error> = Result<(Loc, Tok, Loc), Error>;
|
||||||
pub type ParseError = lalrpop_util::ParseError<usize, Token, HtmlParseError>;
|
|
||||||
|
|
||||||
#[derive(Clone, Debug)]
|
#[derive(Clone, Debug)]
|
||||||
pub enum Token {
|
pub enum Token {
|
||||||
|
@ -107,75 +104,6 @@ pub fn keywordise(tokens: Vec<Token>) -> Vec<Token> {
|
||||||
.collect()
|
.collect()
|
||||||
}
|
}
|
||||||
|
|
||||||
#[derive(Debug)]
|
|
||||||
pub struct HtmlParseError {
|
|
||||||
pub token: Token,
|
|
||||||
pub message: String,
|
|
||||||
}
|
|
||||||
|
|
||||||
fn pprint_token(token: &str) -> &str {
|
|
||||||
match token {
|
|
||||||
"BraceGroupToken" => "code block",
|
|
||||||
"LiteralToken" => "literal",
|
|
||||||
"IdentToken" => "identifier",
|
|
||||||
a => a,
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
fn pprint_tokens(tokens: &[String]) -> String {
|
|
||||||
let tokens: Vec<&str> = tokens.iter().map(|s| pprint_token(&s)).collect();
|
|
||||||
if tokens.len() > 1 {
|
|
||||||
let start = tokens[..tokens.len() - 1].join(", ");
|
|
||||||
let end = &tokens[tokens.len() - 1];
|
|
||||||
format!("{} or {}", start, end)
|
|
||||||
} else {
|
|
||||||
tokens[0].to_string()
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
fn is_in_node_position(tokens: &[String]) -> bool {
|
|
||||||
use std::collections::HashSet;
|
|
||||||
let input: HashSet<&str> = tokens.iter().map(String::as_str).collect();
|
|
||||||
let output: HashSet<&str> = ["\"<\"", "BraceGroupToken", "LiteralToken"]
|
|
||||||
.iter()
|
|
||||||
.cloned()
|
|
||||||
.collect();
|
|
||||||
input == output
|
|
||||||
}
|
|
||||||
|
|
||||||
pub fn parse_error(input: &[Token], error: &ParseError) -> Diagnostic {
|
|
||||||
match error {
|
|
||||||
InvalidToken { location } => {
|
|
||||||
let loc = &input[*location];
|
|
||||||
Diagnostic::spanned(loc.span(), Level::Error, "invalid token")
|
|
||||||
}
|
|
||||||
UnrecognizedToken {
|
|
||||||
token: None,
|
|
||||||
expected,
|
|
||||||
} => panic!(
|
|
||||||
"unexpected end of macro: expecting {}",
|
|
||||||
pprint_tokens(&expected)
|
|
||||||
),
|
|
||||||
UnrecognizedToken {
|
|
||||||
token: Some((_, token, _)),
|
|
||||||
expected,
|
|
||||||
} => {
|
|
||||||
let mut msg = format!("expected {}", pprint_tokens(&expected));
|
|
||||||
if is_in_node_position(expected) && token.is_ident() {
|
|
||||||
// special case: you probably meant to quote that text
|
|
||||||
msg += "; looks like you forgot to put \"quotes\" around your text nodes";
|
|
||||||
}
|
|
||||||
Diagnostic::spanned(token.span(), Level::Error, msg)
|
|
||||||
}
|
|
||||||
ExtraToken {
|
|
||||||
token: (_, token, _),
|
|
||||||
} => Diagnostic::spanned(token.span(), Level::Error, "superfluous token"),
|
|
||||||
User { error } => {
|
|
||||||
Diagnostic::spanned(error.token.span(), Level::Error, error.message.to_owned())
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
pub fn to_stream<I: IntoIterator<Item = Token>>(tokens: I) -> TokenStream {
|
pub fn to_stream<I: IntoIterator<Item = Token>>(tokens: I) -> TokenStream {
|
||||||
let mut stream = TokenStream::new();
|
let mut stream = TokenStream::new();
|
||||||
stream.extend(tokens.into_iter().map(TokenTree::from));
|
stream.extend(tokens.into_iter().map(TokenTree::from));
|
||||||
|
|
|
@ -4,14 +4,15 @@
|
||||||
#![feature(proc_macro_diagnostic)]
|
#![feature(proc_macro_diagnostic)]
|
||||||
#![feature(proc_macro_raw_ident)]
|
#![feature(proc_macro_raw_ident)]
|
||||||
|
|
||||||
|
extern crate ansi_term;
|
||||||
extern crate lalrpop_util;
|
extern crate lalrpop_util;
|
||||||
extern crate pom;
|
|
||||||
extern crate proc_macro;
|
extern crate proc_macro;
|
||||||
|
|
||||||
use proc_macro::{TokenStream, TokenTree};
|
use proc_macro::{quote, TokenStream};
|
||||||
|
|
||||||
mod config;
|
mod config;
|
||||||
mod declare;
|
mod declare;
|
||||||
|
mod error;
|
||||||
mod html;
|
mod html;
|
||||||
mod lexer;
|
mod lexer;
|
||||||
mod map;
|
mod map;
|
||||||
|
@ -22,22 +23,22 @@ pub fn html(input: TokenStream) -> TokenStream {
|
||||||
let stream = lexer::unroll_stream(input, false);
|
let stream = lexer::unroll_stream(input, false);
|
||||||
let result = html::expand_html(&stream);
|
let result = html::expand_html(&stream);
|
||||||
match result {
|
match result {
|
||||||
Err(error) => {
|
Err(err) => {
|
||||||
lexer::parse_error(&stream, &error).emit();
|
error::parse_error(&stream, &err).emit();
|
||||||
panic!("macro expansion produced errors; see above.")
|
quote!(panic!())
|
||||||
}
|
}
|
||||||
Ok(node) => node.into_token_stream(),
|
Ok(node) => node.into_token_stream(),
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
#[proc_macro]
|
#[proc_macro]
|
||||||
pub fn declalrpop_element(input: TokenStream) -> TokenStream {
|
pub fn declare_elements(input: TokenStream) -> TokenStream {
|
||||||
let stream = lexer::keywordise(lexer::unroll_stream(input, true));
|
let stream = lexer::keywordise(lexer::unroll_stream(input, true));
|
||||||
let result = declare::expand_declare_lalrpop(&stream);
|
let result = declare::expand_declare(&stream);
|
||||||
match result {
|
match result {
|
||||||
Err(error) => {
|
Err(err) => {
|
||||||
lexer::parse_error(&stream, &error).emit();
|
error::parse_error(&stream, &err).emit();
|
||||||
panic!("macro expansion produced errors; see above.")
|
quote!(panic!())
|
||||||
}
|
}
|
||||||
Ok(decls) => {
|
Ok(decls) => {
|
||||||
let mut out = TokenStream::new();
|
let mut out = TokenStream::new();
|
||||||
|
@ -48,16 +49,3 @@ pub fn declalrpop_element(input: TokenStream) -> TokenStream {
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
#[proc_macro]
|
|
||||||
pub fn declare_element(input: TokenStream) -> TokenStream {
|
|
||||||
let input: Vec<TokenTree> = input.into_iter().collect();
|
|
||||||
let result = declare::expand_declare(&input);
|
|
||||||
match result {
|
|
||||||
Err(error) => {
|
|
||||||
parser::parse_error(&input, &error).emit();
|
|
||||||
panic!("macro expansion produced errors; see above.")
|
|
||||||
}
|
|
||||||
Ok(ts) => ts,
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
|
@ -1,89 +1,3 @@
|
||||||
use lalrpop_util::lalrpop_mod;
|
use lalrpop_util::lalrpop_mod;
|
||||||
use pom::combinator::*;
|
|
||||||
use pom::{Error, Parser};
|
|
||||||
use proc_macro::{Diagnostic, Group, Ident, Level, Punct, TokenStream, TokenTree};
|
|
||||||
|
|
||||||
lalrpop_mod!(pub grammar);
|
lalrpop_mod!(pub grammar);
|
||||||
|
|
||||||
pub fn punct<'a>(punct: char) -> Combinator<impl Parser<'a, TokenTree, Output = Punct>> {
|
|
||||||
comb(move |input: &[TokenTree], start| match input.get(start) {
|
|
||||||
Some(TokenTree::Punct(p)) if p.as_char() == punct => Ok((p.clone(), start + 1)),
|
|
||||||
_ => Err(Error::Mismatch {
|
|
||||||
message: format!("expected {:?}", punct),
|
|
||||||
position: start,
|
|
||||||
}),
|
|
||||||
})
|
|
||||||
}
|
|
||||||
|
|
||||||
pub fn ident<'a>() -> Combinator<impl Parser<'a, TokenTree, Output = Ident>> {
|
|
||||||
comb(|input: &[TokenTree], start| match input.get(start) {
|
|
||||||
Some(TokenTree::Ident(i)) => Ok((i.clone(), start + 1)),
|
|
||||||
_ => Err(Error::Mismatch {
|
|
||||||
message: "expected identifier".to_string(),
|
|
||||||
position: start,
|
|
||||||
}),
|
|
||||||
})
|
|
||||||
}
|
|
||||||
|
|
||||||
pub fn group<'a>() -> Combinator<impl Parser<'a, TokenTree, Output = Group>> {
|
|
||||||
comb(|input: &[TokenTree], start| match input.get(start) {
|
|
||||||
Some(TokenTree::Group(g)) => Ok((g.clone(), start + 1)),
|
|
||||||
_ => Err(Error::Mismatch {
|
|
||||||
message: "expected group".to_string(),
|
|
||||||
position: start,
|
|
||||||
}),
|
|
||||||
})
|
|
||||||
}
|
|
||||||
|
|
||||||
fn to_stream<'a, I: IntoIterator<Item = &'a TokenTree>>(tokens: I) -> TokenStream {
|
|
||||||
let mut stream = TokenStream::new();
|
|
||||||
stream.extend(tokens.into_iter().cloned());
|
|
||||||
stream
|
|
||||||
}
|
|
||||||
|
|
||||||
pub fn type_spec<'a>() -> Combinator<impl Parser<'a, TokenTree, Output = TokenStream>> {
|
|
||||||
let valid = ident().map(TokenTree::Ident)
|
|
||||||
| punct(':').map(TokenTree::Punct)
|
|
||||||
| punct('<').map(TokenTree::Punct)
|
|
||||||
| punct('>').map(TokenTree::Punct)
|
|
||||||
| punct('&').map(TokenTree::Punct)
|
|
||||||
| punct('\'').map(TokenTree::Punct);
|
|
||||||
valid.repeat(1..).collect().map(to_stream)
|
|
||||||
}
|
|
||||||
|
|
||||||
/// Turn a parser error into a proc_macro diagnostic.
|
|
||||||
pub fn parse_error(input: &[TokenTree], error: &pom::Error) -> Diagnostic {
|
|
||||||
match error {
|
|
||||||
pom::Error::Incomplete => Diagnostic::new(Level::Error, "unexpected end of macro!"),
|
|
||||||
pom::Error::Mismatch { message, position } => {
|
|
||||||
Diagnostic::spanned(input[*position].span(), Level::Error, message.as_str())
|
|
||||||
}
|
|
||||||
pom::Error::Conversion { message, position } => {
|
|
||||||
Diagnostic::spanned(input[*position].span(), Level::Error, message.as_str())
|
|
||||||
}
|
|
||||||
pom::Error::Expect {
|
|
||||||
message,
|
|
||||||
position,
|
|
||||||
inner,
|
|
||||||
} => {
|
|
||||||
let mut diag =
|
|
||||||
Diagnostic::spanned(input[*position].span(), Level::Error, message.as_str());
|
|
||||||
let child = parse_error(input, &inner);
|
|
||||||
diag.span_error(child.spans(), child.message())
|
|
||||||
}
|
|
||||||
pom::Error::Custom {
|
|
||||||
message,
|
|
||||||
position,
|
|
||||||
inner,
|
|
||||||
} => {
|
|
||||||
let mut diag =
|
|
||||||
Diagnostic::spanned(input[*position].span(), Level::Error, message.as_str());
|
|
||||||
if let Some(inner) = inner {
|
|
||||||
let child = parse_error(input, &inner);
|
|
||||||
diag.span_error(child.spans(), child.message())
|
|
||||||
} else {
|
|
||||||
diag
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
|
@ -2,7 +2,7 @@
|
||||||
#![allow(dead_code)]
|
#![allow(dead_code)]
|
||||||
|
|
||||||
use std::fmt::Display;
|
use std::fmt::Display;
|
||||||
use typed_html_macros::{declalrpop_element, declare_element};
|
use typed_html_macros::declare_elements;
|
||||||
|
|
||||||
use super::types::*;
|
use super::types::*;
|
||||||
|
|
||||||
|
@ -99,7 +99,7 @@ impl Node for TextNode {
|
||||||
impl FlowContent for TextNode {}
|
impl FlowContent for TextNode {}
|
||||||
impl PhrasingContent for TextNode {}
|
impl PhrasingContent for TextNode {}
|
||||||
|
|
||||||
declalrpop_element!{
|
declare_elements!{
|
||||||
html {
|
html {
|
||||||
xmlns: Uri,
|
xmlns: Uri,
|
||||||
} with [head, body];
|
} with [head, body];
|
||||||
|
@ -196,25 +196,22 @@ declalrpop_element!{
|
||||||
details {
|
details {
|
||||||
open: bool,
|
open: bool,
|
||||||
} in [FlowContent, SectioningContent, InteractiveContent] with [summary] FlowContent;
|
} in [FlowContent, SectioningContent, InteractiveContent] with [summary] FlowContent;
|
||||||
}
|
dfn in [FlowContent, PhrasingContent] with PhrasingContent;
|
||||||
|
div in [FlowContent] with FlowContent;
|
||||||
// Flow content
|
dl in [FlowContent] with DescriptionListContent;
|
||||||
declare_element!(dfn {} [] [FlowContent, PhrasingContent] PhrasingContent);
|
em in [FlowContent, PhrasingContent] with PhrasingContent;
|
||||||
declare_element!(div {} [] [FlowContent] FlowContent);
|
embed {
|
||||||
declare_element!(dl {} [] [FlowContent] DescriptionListContent);
|
|
||||||
declare_element!(em {} [] [FlowContent, PhrasingContent] PhrasingContent);
|
|
||||||
declare_element!(embed {
|
|
||||||
height: usize,
|
height: usize,
|
||||||
src: Uri,
|
src: Uri,
|
||||||
type: Mime,
|
type: Mime,
|
||||||
width: usize,
|
width: usize,
|
||||||
} [] [FlowContent, PhrasingContent, EmbeddedContent, InteractiveContent]);
|
} in [FlowContent, PhrasingContent, EmbeddedContent, InteractiveContent];
|
||||||
// FIXME the legend attribute should be optional
|
// FIXME the legend attribute should be optional
|
||||||
declare_element!(fieldset {} [legend] [FlowContent, SectioningContent, FormContent] FlowContent);
|
fieldset in [FlowContent, SectioningContent, FormContent] with [legend] FlowContent;
|
||||||
// FIXME the figcaption attribute should be optional
|
// FIXME the figcaption attribute should be optional
|
||||||
declare_element!(figure {} [figcaption] [FlowContent, SectioningContent] FlowContent);
|
figure in [FlowContent, SectioningContent] with [figcaption] FlowContent;
|
||||||
declare_element!(footer {} [] [FlowContent] FlowContent);
|
footer in [FlowContent] with FlowContent;
|
||||||
declare_element!(form {
|
form {
|
||||||
accept-charset: SpacedList<CharacterEncoding>,
|
accept-charset: SpacedList<CharacterEncoding>,
|
||||||
action: Uri,
|
action: Uri,
|
||||||
autocomplete: OnOff,
|
autocomplete: OnOff,
|
||||||
|
@ -223,18 +220,18 @@ declare_element!(form {
|
||||||
name: Id,
|
name: Id,
|
||||||
novalidate: bool,
|
novalidate: bool,
|
||||||
target: Target,
|
target: Target,
|
||||||
} [] [FlowContent] FlowContent);
|
} in [FlowContent] with FlowContent;
|
||||||
declare_element!(h1 {} [] [FlowContent, HeadingContent, HGroupContent] PhrasingContent);
|
h1 in [FlowContent, HeadingContent, HGroupContent] with PhrasingContent;
|
||||||
declare_element!(h2 {} [] [FlowContent, HeadingContent, HGroupContent] PhrasingContent);
|
h2 in [FlowContent, HeadingContent, HGroupContent] with PhrasingContent;
|
||||||
declare_element!(h3 {} [] [FlowContent, HeadingContent, HGroupContent] PhrasingContent);
|
h3 in [FlowContent, HeadingContent, HGroupContent] with PhrasingContent;
|
||||||
declare_element!(h4 {} [] [FlowContent, HeadingContent, HGroupContent] PhrasingContent);
|
h4 in [FlowContent, HeadingContent, HGroupContent] with PhrasingContent;
|
||||||
declare_element!(h5 {} [] [FlowContent, HeadingContent, HGroupContent] PhrasingContent);
|
h5 in [FlowContent, HeadingContent, HGroupContent] with PhrasingContent;
|
||||||
declare_element!(h6 {} [] [FlowContent, HeadingContent, HGroupContent] PhrasingContent);
|
h6 in [FlowContent, HeadingContent, HGroupContent] with PhrasingContent;
|
||||||
declare_element!(header {} [] [FlowContent] FlowContent);
|
header in [FlowContent] with FlowContent;
|
||||||
declare_element!(hgroup {} [] [FlowContent, HeadingContent] HGroupContent);
|
hgroup in [FlowContent, HeadingContent] with HGroupContent;
|
||||||
declare_element!(hr {} [] [FlowContent]);
|
hr in [FlowContent];
|
||||||
declare_element!(i {} [] [FlowContent, PhrasingContent] PhrasingContent);
|
i in [FlowContent, PhrasingContent] with PhrasingContent;
|
||||||
declare_element!(iframe {
|
iframe {
|
||||||
allow: FeaturePolicy,
|
allow: FeaturePolicy,
|
||||||
allowfullscreen: bool,
|
allowfullscreen: bool,
|
||||||
allowpaymentrequest: bool,
|
allowpaymentrequest: bool,
|
||||||
|
@ -245,8 +242,8 @@ declare_element!(iframe {
|
||||||
src: Uri,
|
src: Uri,
|
||||||
srcdoc: Uri,
|
srcdoc: Uri,
|
||||||
width: usize,
|
width: usize,
|
||||||
} [] [FlowContent, PhrasingContent, EmbeddedContent, InteractiveContent] FlowContent);
|
} in [FlowContent, PhrasingContent, EmbeddedContent, InteractiveContent] with FlowContent;
|
||||||
declare_element!(img {
|
img {
|
||||||
alt: String,
|
alt: String,
|
||||||
crossorigin: CrossOrigin,
|
crossorigin: CrossOrigin,
|
||||||
decoding: ImageDecoding,
|
decoding: ImageDecoding,
|
||||||
|
@ -257,8 +254,8 @@ declare_element!(img {
|
||||||
srcset: String, // FIXME this is much more complicated
|
srcset: String, // FIXME this is much more complicated
|
||||||
usemap: String, // FIXME should be a fragment starting with '#'
|
usemap: String, // FIXME should be a fragment starting with '#'
|
||||||
width: usize,
|
width: usize,
|
||||||
} [] [FlowContent, PhrasingContent, EmbeddedContent]);
|
} in [FlowContent, PhrasingContent, EmbeddedContent];
|
||||||
declare_element!(input {
|
input {
|
||||||
autocomplete: String,
|
autocomplete: String,
|
||||||
autofocus: bool,
|
autofocus: bool,
|
||||||
disabled: bool,
|
disabled: bool,
|
||||||
|
@ -269,23 +266,23 @@ declare_element!(input {
|
||||||
tabindex: usize,
|
tabindex: usize,
|
||||||
type: InputType,
|
type: InputType,
|
||||||
value: String,
|
value: String,
|
||||||
} [] [FlowContent, FormContent, PhrasingContent]);
|
} in [FlowContent, FormContent, PhrasingContent];
|
||||||
declare_element!(ins {
|
ins {
|
||||||
cite: Uri,
|
cite: Uri,
|
||||||
datetime: Datetime,
|
datetime: Datetime,
|
||||||
} [] [FlowContent, PhrasingContent] FlowContent);
|
} in [FlowContent, PhrasingContent] with FlowContent;
|
||||||
declare_element!(kbd {} [] [FlowContent, PhrasingContent] PhrasingContent);
|
kbd in [FlowContent, PhrasingContent] with PhrasingContent;
|
||||||
declare_element!(label {
|
label {
|
||||||
for: Id,
|
for: Id,
|
||||||
form: Id,
|
form: Id,
|
||||||
} [] [FlowContent, PhrasingContent, InteractiveContent, FormContent] PhrasingContent);
|
} in [FlowContent, PhrasingContent, InteractiveContent, FormContent] with PhrasingContent;
|
||||||
declare_element!(main {} [] [FlowContent] FlowContent);
|
main in [FlowContent] with FlowContent;
|
||||||
declare_element!(map {
|
map {
|
||||||
name: Id,
|
name: Id,
|
||||||
} [] [FlowContent, PhrasingContent] MapContent);
|
} in [FlowContent, PhrasingContent] with MapContent;
|
||||||
declare_element!(mark {} [] [FlowContent, PhrasingContent] PhrasingContent);
|
mark in [FlowContent, PhrasingContent] with PhrasingContent;
|
||||||
// TODO the <math> element
|
// TODO the <math> element
|
||||||
declare_element!(meter {
|
meter {
|
||||||
value: isize,
|
value: isize,
|
||||||
min: isize,
|
min: isize,
|
||||||
max: isize,
|
max: isize,
|
||||||
|
@ -293,10 +290,10 @@ declare_element!(meter {
|
||||||
high: isize,
|
high: isize,
|
||||||
optimum: isize,
|
optimum: isize,
|
||||||
form: Id,
|
form: Id,
|
||||||
} [] [FlowContent, PhrasingContent] PhrasingContent);
|
} in [FlowContent, PhrasingContent] with PhrasingContent;
|
||||||
declare_element!(nav {} [] [FlowContent, SectioningContent] PhrasingContent);
|
nav in [FlowContent, SectioningContent] with PhrasingContent;
|
||||||
declare_element!(noscript {} [] [MetadataContent, FlowContent, PhrasingContent] Node);
|
noscript in [MetadataContent, FlowContent, PhrasingContent] with Node;
|
||||||
declare_element!(object {
|
object {
|
||||||
data: Uri,
|
data: Uri,
|
||||||
form: Id,
|
form: Id,
|
||||||
height: usize,
|
height: usize,
|
||||||
|
@ -305,30 +302,30 @@ declare_element!(object {
|
||||||
typemustmatch: bool,
|
typemustmatch: bool,
|
||||||
usemap: String, // TODO should be a fragment starting with '#'
|
usemap: String, // TODO should be a fragment starting with '#'
|
||||||
width: usize,
|
width: usize,
|
||||||
} [] [FlowContent, PhrasingContent, EmbeddedContent, InteractiveContent, FormContent] Element_param);
|
} in [FlowContent, PhrasingContent, EmbeddedContent, InteractiveContent, FormContent] with Element_param;
|
||||||
declare_element!(ol {
|
ol {
|
||||||
reversed: bool,
|
reversed: bool,
|
||||||
start: isize,
|
start: isize,
|
||||||
type: OrderedListType,
|
type: OrderedListType,
|
||||||
} [] [FlowContent] Element_li);
|
} in [FlowContent] with Element_li;
|
||||||
declare_element!(output {
|
output {
|
||||||
for: SpacedSet<Id>,
|
for: SpacedSet<Id>,
|
||||||
form: Id,
|
form: Id,
|
||||||
name: Id,
|
name: Id,
|
||||||
} [] [FlowContent, PhrasingContent, FormContent] PhrasingContent);
|
} in [FlowContent, PhrasingContent, FormContent] with PhrasingContent;
|
||||||
declare_element!(p {} [] [FlowContent] PhrasingContent);
|
p in [FlowContent] with PhrasingContent;
|
||||||
declare_element!(pre {} [] [FlowContent] PhrasingContent);
|
pre in [FlowContent] with PhrasingContent;
|
||||||
declare_element!(progress {
|
progress {
|
||||||
max: f64,
|
max: f64,
|
||||||
value: f64,
|
value: f64,
|
||||||
} [] [FlowContent, PhrasingContent] PhrasingContent);
|
} in [FlowContent, PhrasingContent] with PhrasingContent;
|
||||||
declare_element!(q {
|
q {
|
||||||
cite: Uri,
|
cite: Uri,
|
||||||
} [] [FlowContent, PhrasingContent] PhrasingContent);
|
} in [FlowContent, PhrasingContent] with PhrasingContent;
|
||||||
declare_element!(ruby {} [] [FlowContent, PhrasingContent] PhrasingContent);
|
ruby in [FlowContent, PhrasingContent] with PhrasingContent;
|
||||||
declare_element!(s {} [] [FlowContent, PhrasingContent] PhrasingContent);
|
s in [FlowContent, PhrasingContent] with PhrasingContent;
|
||||||
declare_element!(samp {} [] [FlowContent, PhrasingContent] PhrasingContent);
|
samp in [FlowContent, PhrasingContent] with PhrasingContent;
|
||||||
declare_element!(script {
|
script {
|
||||||
async: bool,
|
async: bool,
|
||||||
crossorigin: CrossOrigin,
|
crossorigin: CrossOrigin,
|
||||||
defer: bool,
|
defer: bool,
|
||||||
|
@ -338,9 +335,9 @@ declare_element!(script {
|
||||||
src: Uri,
|
src: Uri,
|
||||||
text: String,
|
text: String,
|
||||||
type: String, // TODO could be an enum
|
type: String, // TODO could be an enum
|
||||||
} [] [MetadataContent, FlowContent, PhrasingContent, TableColumnContent] TextNode);
|
} in [MetadataContent, FlowContent, PhrasingContent, TableColumnContent] with TextNode;
|
||||||
declare_element!(section {} [] [FlowContent, SectioningContent] FlowContent);
|
section in [FlowContent, SectioningContent] with FlowContent;
|
||||||
declare_element!(select {
|
select {
|
||||||
autocomplete: String,
|
autocomplete: String,
|
||||||
autofocus: bool,
|
autofocus: bool,
|
||||||
disabled: bool,
|
disabled: bool,
|
||||||
|
@ -349,16 +346,15 @@ declare_element!(select {
|
||||||
name: Id,
|
name: Id,
|
||||||
required: bool,
|
required: bool,
|
||||||
size: usize,
|
size: usize,
|
||||||
} [] [FlowContent, PhrasingContent, InteractiveContent, FormContent] SelectContent);
|
} in [FlowContent, PhrasingContent, InteractiveContent, FormContent] with SelectContent;
|
||||||
declare_element!(small {} [] [FlowContent, PhrasingContent] PhrasingContent);
|
small in [FlowContent, PhrasingContent] with PhrasingContent;
|
||||||
declare_element!(span {} [] [FlowContent, PhrasingContent] PhrasingContent);
|
span in [FlowContent, PhrasingContent] with PhrasingContent;
|
||||||
declare_element!(strong {} [] [FlowContent, PhrasingContent] PhrasingContent);
|
strong in [FlowContent, PhrasingContent] with PhrasingContent;
|
||||||
declare_element!(sub {} [] [FlowContent, PhrasingContent] PhrasingContent);
|
sub in [FlowContent, PhrasingContent] with PhrasingContent;
|
||||||
declare_element!(sup {} [] [FlowContent, PhrasingContent] PhrasingContent);
|
sup in [FlowContent, PhrasingContent] with PhrasingContent;
|
||||||
// TODO the <svg> element
|
table in [FlowContent] with TableContent;
|
||||||
declare_element!(table {} [] [FlowContent] TableContent);
|
template in [MetadataContent, FlowContent, PhrasingContent, TableColumnContent] with Node;
|
||||||
declare_element!(template {} [] [MetadataContent, FlowContent, PhrasingContent, TableColumnContent] Node);
|
textarea {
|
||||||
declare_element!(textarea {
|
|
||||||
autocomplete: OnOff,
|
autocomplete: OnOff,
|
||||||
autofocus: bool,
|
autofocus: bool,
|
||||||
cols: usize,
|
cols: usize,
|
||||||
|
@ -373,17 +369,17 @@ declare_element!(textarea {
|
||||||
rows: usize,
|
rows: usize,
|
||||||
spellcheck: BoolOrDefault,
|
spellcheck: BoolOrDefault,
|
||||||
wrap: Wrap,
|
wrap: Wrap,
|
||||||
} [] [FlowContent, PhrasingContent, InteractiveContent, FormContent] TextNode);
|
} in [FlowContent, PhrasingContent, InteractiveContent, FormContent] with TextNode;
|
||||||
declare_element!(time {
|
time {
|
||||||
datetime: Datetime,
|
datetime: Datetime,
|
||||||
} [] [FlowContent, PhrasingContent] PhrasingContent);
|
} in [FlowContent, PhrasingContent] with PhrasingContent;
|
||||||
declare_element!(ul {} [] [FlowContent] Element_li);
|
ul in [FlowContent] with Element_li;
|
||||||
declare_element!(var {} [] [FlowContent, PhrasingContent] PhrasingContent);
|
var in [FlowContent, PhrasingContent] with PhrasingContent;
|
||||||
declare_element!(video {} [] [FlowContent, PhrasingContent, EmbeddedContent] MediaContent);
|
video in [FlowContent, PhrasingContent, EmbeddedContent] with MediaContent;
|
||||||
declare_element!(wbr {} [] [FlowContent, PhrasingContent]);
|
wbr in [FlowContent, PhrasingContent];
|
||||||
|
|
||||||
// Non-group elements
|
// Non-group elements
|
||||||
declare_element!(area {
|
area {
|
||||||
alt: String,
|
alt: String,
|
||||||
coords: String, // TODO could perhaps be validated
|
coords: String, // TODO could perhaps be validated
|
||||||
download: bool,
|
download: bool,
|
||||||
|
@ -393,67 +389,67 @@ declare_element!(area {
|
||||||
rel: SpacedSet<LinkType>,
|
rel: SpacedSet<LinkType>,
|
||||||
shape: AreaShape,
|
shape: AreaShape,
|
||||||
target: Target,
|
target: Target,
|
||||||
} [] [MapContent]);
|
} in [MapContent];
|
||||||
declare_element!(caption {} [] [TableContent] FlowContent);
|
caption in [TableContent] with FlowContent;
|
||||||
declare_element!(col {
|
col {
|
||||||
span: usize,
|
span: usize,
|
||||||
} [] []);
|
};
|
||||||
declare_element!(colgroup {
|
colgroup {
|
||||||
span: usize,
|
span: usize,
|
||||||
} [] [TableContent] Element_col);
|
} in [TableContent] with Element_col;
|
||||||
declare_element!(dd {} [] [DescriptionListContent] FlowContent);
|
dd in [DescriptionListContent] with FlowContent;
|
||||||
declare_element!(dt {} [] [DescriptionListContent] FlowContent);
|
dt in [DescriptionListContent] with FlowContent;
|
||||||
declare_element!(figcaption {} [] [] FlowContent);
|
figcaption with FlowContent;
|
||||||
declare_element!(legend {} [] [] PhrasingContent);
|
legend with PhrasingContent;
|
||||||
declare_element!(li {
|
li {
|
||||||
value: isize,
|
value: isize,
|
||||||
} [] [] FlowContent);
|
} with FlowContent;
|
||||||
declare_element!(option {
|
option {
|
||||||
disabled: bool,
|
disabled: bool,
|
||||||
label: String,
|
label: String,
|
||||||
selected: bool,
|
selected: bool,
|
||||||
value: String,
|
value: String,
|
||||||
} [] [SelectContent] TextNode);
|
} in [SelectContent] with TextNode;
|
||||||
declare_element!(optgroup {
|
optgroup {
|
||||||
disabled: bool,
|
disabled: bool,
|
||||||
label: String,
|
label: String,
|
||||||
} [] [SelectContent] Element_option);
|
} in [SelectContent] with Element_option;
|
||||||
declare_element!(param {
|
param {
|
||||||
name: String,
|
name: String,
|
||||||
value: String,
|
value: String,
|
||||||
} [] []);
|
};
|
||||||
declare_element!(source {
|
source {
|
||||||
src: Uri,
|
src: Uri,
|
||||||
type: Mime,
|
type: Mime,
|
||||||
} [] [MediaContent]);
|
} in [MediaContent];
|
||||||
declare_element!(summary {} [] [] PhrasingContent);
|
summary with PhrasingContent;
|
||||||
declare_element!(tbody {} [] [TableContent] Element_tr);
|
tbody in [TableContent] with Element_tr;
|
||||||
declare_element!(td {
|
td {
|
||||||
colspan: usize,
|
colspan: usize,
|
||||||
headers: SpacedSet<Id>,
|
headers: SpacedSet<Id>,
|
||||||
rowspan: usize,
|
rowspan: usize,
|
||||||
} [] [TableColumnContent] FlowContent);
|
} in [TableColumnContent] with FlowContent;
|
||||||
declare_element!(tfoot {} [] [TableContent] Element_tr);
|
tfoot in [TableContent] with Element_tr;
|
||||||
declare_element!(th {
|
th {
|
||||||
abbr: String,
|
abbr: String,
|
||||||
colspan: usize,
|
colspan: usize,
|
||||||
headers: SpacedSet<Id>,
|
headers: SpacedSet<Id>,
|
||||||
rowspan: usize,
|
rowspan: usize,
|
||||||
scope: TableHeaderScope,
|
scope: TableHeaderScope,
|
||||||
} [] [TableColumnContent] FlowContent);
|
} in [TableColumnContent] with FlowContent;
|
||||||
declare_element!(thead {} [] [TableContent] Element_tr);
|
thead in [TableContent] with Element_tr;
|
||||||
declare_element!(tr {} [] [TableContent] TableColumnContent);
|
tr in [TableContent] with TableColumnContent;
|
||||||
declare_element!(track {
|
track {
|
||||||
default: bool,
|
default: bool,
|
||||||
kind: VideoKind,
|
kind: VideoKind,
|
||||||
label: String,
|
label: String,
|
||||||
src: Uri,
|
src: Uri,
|
||||||
srclang: LanguageTag,
|
srclang: LanguageTag,
|
||||||
} [] [MediaContent]);
|
} in [MediaContent];
|
||||||
|
|
||||||
// Don't @ me
|
// Don't @ me
|
||||||
declare_element!(blink {} [] [FlowContent, PhrasingContent] PhrasingContent);
|
blink in [FlowContent, PhrasingContent] with PhrasingContent;
|
||||||
declare_element!(marquee {
|
marquee {
|
||||||
behavior: String, // FIXME enum
|
behavior: String, // FIXME enum
|
||||||
bgcolor: String, // FIXME colour
|
bgcolor: String, // FIXME colour
|
||||||
direction: String, // FIXME direction enum
|
direction: String, // FIXME direction enum
|
||||||
|
@ -465,4 +461,5 @@ declare_element!(marquee {
|
||||||
truespeed: bool,
|
truespeed: bool,
|
||||||
vspace: String, // FIXME size
|
vspace: String, // FIXME size
|
||||||
width: String, // FIXME size
|
width: String, // FIXME size
|
||||||
} [] [FlowContent, PhrasingContent] PhrasingContent);
|
} in [FlowContent, PhrasingContent] with PhrasingContent;
|
||||||
|
}
|
||||||
|
|
Loading…
Reference in New Issue