2018-07-12 16:14:29 -07:00
|
|
|
-- Nomsu version 2
|
|
|
|
file (File):
|
|
|
|
{:curr_indent: '' :}
|
|
|
|
blank_line*
|
|
|
|
(chunk (nl_nodent chunk_delimeter nl_nodent chunk)*)?
|
|
|
|
blank_line*
|
|
|
|
(!. / (!! .* -> "Parse error" !!))
|
|
|
|
|
|
|
|
nodent: =curr_indent !(" ")
|
|
|
|
indent: =curr_indent " " !(" ")
|
|
|
|
dedent: !(=curr_indent) (" ")*
|
|
|
|
eol: %ws* (!. / &%nl)
|
|
|
|
|
|
|
|
comment (Comment):
|
|
|
|
"#" {~ [^%nl]* (%nl+ ({:curr_indent: indent :} -> '') [^%nl]* (%nl+ (=curr_indent -> '') [^%nl]*)* (!. / nl_dedent))? ~}
|
|
|
|
inline_comment (Comment):
|
|
|
|
"(#" {~ (inline_comment / [^%nl])* ~} "#)"
|
|
|
|
|
|
|
|
blank_line: %nl eol
|
|
|
|
nl_nodent: blank_line* %nl nodent
|
|
|
|
nl_indent: blank_line* %nl {:curr_indent: indent :}
|
|
|
|
nl_dedent: blank_line* %nl &dedent
|
|
|
|
|
|
|
|
chunk: !chunk_delimeter (block/action/expression)
|
|
|
|
chunk_delimeter: ("~")^+3
|
2018-06-18 15:44:29 -07:00
|
|
|
|
2018-04-25 17:43:48 -07:00
|
|
|
inline_block (Block):
|
2018-07-12 16:14:29 -07:00
|
|
|
(inline_comment / inline_statement) (%ws* ";" %ws* (inline_comment / inline_statement))+
|
2018-04-25 17:43:48 -07:00
|
|
|
block (Block):
|
2018-07-12 16:14:29 -07:00
|
|
|
block_line (nl_nodent block_line)+
|
|
|
|
block_line:
|
|
|
|
comment / inline_comment / statement / (!! [^%nl]* -> "Unexpected character while parsing block line" !!)
|
2017-12-30 14:31:07 -08:00
|
|
|
|
2018-07-12 16:14:29 -07:00
|
|
|
statement: (action / expression) (eol / (!! [^%nl]* -> "Unexpected character while parsing line" !!))
|
2018-05-26 13:09:20 -07:00
|
|
|
inline_statement: inline_action / inline_expression
|
2018-04-06 16:45:51 -07:00
|
|
|
|
|
|
|
noindex_inline_expression:
|
2018-01-11 01:09:26 -08:00
|
|
|
number / variable / inline_text / inline_list / inline_dict / inline_nomsu
|
2018-05-10 22:47:03 -07:00
|
|
|
/ ( "("
|
|
|
|
%ws* (inline_block / inline_action / inline_expression) %ws*
|
2018-06-12 13:56:15 -07:00
|
|
|
(%ws* ',' %ws* (inline_block / inline_action / inline_expression) %ws*)*
|
2018-05-03 16:30:55 -07:00
|
|
|
(")"
|
2018-07-12 16:14:29 -07:00
|
|
|
/ (!! eol -> 'Line ended without finding a closing )-parenthesis' !!)
|
|
|
|
/ (!! [^%nl]+ -> 'Unexpected character while parsing subexpression' !!)
|
2018-05-03 16:30:55 -07:00
|
|
|
)
|
2018-05-10 22:47:03 -07:00
|
|
|
)
|
2018-07-12 16:14:29 -07:00
|
|
|
inline_expression: index_chain / noindex_inline_expression
|
2018-01-19 17:29:44 -08:00
|
|
|
indented_expression:
|
2018-04-25 17:43:48 -07:00
|
|
|
indented_text / indented_nomsu / indented_list / indented_dict
|
2018-07-12 16:14:29 -07:00
|
|
|
/ (("(..)" / ":")? nl_indent
|
2018-05-03 16:30:55 -07:00
|
|
|
(block / action / expression)
|
2018-07-12 16:14:29 -07:00
|
|
|
(!. / &nl_dedent / (!! (!nl_dedent .)* -> "Unexpected character while parsing indented expression" !!))
|
2018-04-25 17:43:48 -07:00
|
|
|
)
|
2018-01-19 17:29:44 -08:00
|
|
|
expression:
|
2018-05-30 13:07:08 -07:00
|
|
|
inline_expression
|
|
|
|
/ (":" %ws* ((inline_block / inline_action / inline_expression) eol
|
2018-07-12 16:14:29 -07:00
|
|
|
/ (!! eol -> "Missing expression after the ':'" !!)))
|
2018-05-30 13:07:08 -07:00
|
|
|
/ indented_expression
|
2017-12-30 14:31:07 -08:00
|
|
|
|
2018-07-12 16:14:29 -07:00
|
|
|
inline_nomsu (EscapedNomsu): "\" inline_expression
|
2018-05-26 13:09:20 -07:00
|
|
|
indented_nomsu (EscapedNomsu):
|
2018-07-12 16:14:29 -07:00
|
|
|
"\" (
|
2018-05-30 13:07:08 -07:00
|
|
|
noindex_inline_expression
|
|
|
|
/ (":" %ws* ((inline_block / inline_action / inline_expression) eol
|
2018-07-12 16:14:29 -07:00
|
|
|
/ (!! eol -> "Missing expression after the ':'" !!)))
|
|
|
|
/ indented_expression)
|
2018-05-26 13:09:20 -07:00
|
|
|
|
|
|
|
index_chain (IndexChain):
|
2018-07-12 16:14:29 -07:00
|
|
|
noindex_inline_expression ("." (text_word / noindex_inline_expression))+
|
2018-05-26 13:09:20 -07:00
|
|
|
|
2018-06-12 13:56:15 -07:00
|
|
|
-- Actions need either at least 1 word, or at least 2 tokens
|
2018-04-17 14:18:23 -07:00
|
|
|
inline_action (Action):
|
2018-06-12 13:56:15 -07:00
|
|
|
( (inline_expression (%ws* (inline_expression / word))+)
|
|
|
|
/ (word (%ws* (inline_expression / word))*))
|
2018-05-30 13:07:08 -07:00
|
|
|
(%ws* ":" %ws* (inline_block / inline_action / inline_expression
|
2018-07-12 16:14:29 -07:00
|
|
|
/ (!! '' -> "Missing expression after the ':'" !!)))?
|
2018-04-17 14:18:23 -07:00
|
|
|
action (Action):
|
2018-07-12 16:14:29 -07:00
|
|
|
(expression ((nl_nodent "..")? %ws* (expression / word))+)
|
|
|
|
/ (word ((nl_nodent "..")? %ws* (expression / word))*)
|
2017-12-30 14:31:07 -08:00
|
|
|
|
2018-07-12 16:14:29 -07:00
|
|
|
word: !number ( %operator_char+ / %ident_char+ )
|
2017-12-30 14:31:07 -08:00
|
|
|
|
2018-07-12 16:14:29 -07:00
|
|
|
text_word (Text): word
|
2018-05-10 22:47:03 -07:00
|
|
|
|
2018-01-11 01:09:26 -08:00
|
|
|
inline_text (Text):
|
2018-01-30 15:10:37 -08:00
|
|
|
!('".."' eol)
|
2018-07-12 16:14:29 -07:00
|
|
|
'"'
|
2018-01-26 14:53:02 -08:00
|
|
|
({~ (('\"' -> '"') / ('\\' -> '\') / %escaped_char / [^%nl\"])+ ~}
|
2018-01-30 15:13:04 -08:00
|
|
|
/ inline_text_interpolation)*
|
2018-07-12 16:14:29 -07:00
|
|
|
('"'
|
|
|
|
/ (!! eol -> 'Line ended before finding a closing double quotation mark' !!)
|
|
|
|
/ (!! [^%nl]* -> 'Unexpected character while parsing Text' !!))
|
2018-01-30 15:13:04 -08:00
|
|
|
inline_text_interpolation:
|
2018-01-30 15:10:37 -08:00
|
|
|
"\" (
|
|
|
|
variable / inline_list / inline_dict / inline_text
|
2018-05-03 16:30:55 -07:00
|
|
|
/ ("("
|
|
|
|
%ws* (inline_block / inline_action / inline_expression) %ws*
|
2018-06-12 13:56:15 -07:00
|
|
|
(%ws* ',' %ws* (inline_block / inline_action / inline_expression) %ws*)*
|
2018-05-03 16:30:55 -07:00
|
|
|
(")"
|
2018-07-12 16:14:29 -07:00
|
|
|
/ (!! &%nl -> 'Line ended without finding a closing )-parenthesis' !!)
|
|
|
|
/ (!! [^%nl]* -> 'Unexpected character while parsing Text interpolation' !!)))
|
2018-01-30 15:13:04 -08:00
|
|
|
)
|
2018-07-10 17:10:37 -07:00
|
|
|
|
|
|
|
-- Have to use "%indent" instead of "indent" etc. to avoid messing up text lines that start with "#"
|
|
|
|
indented_text (Text):
|
2018-07-12 16:14:29 -07:00
|
|
|
'".."' eol %nl
|
|
|
|
{~ (%nl*) ({:curr_indent: indent :} -> "") ~}
|
|
|
|
(indented_plain_text / text_interpolation / {~ %nl+ (nodent -> "") ~})*
|
|
|
|
(!. / &nl_dedent / (!! (!nl_dedent .)* -> "Unexpected character while parsing Text" !!))
|
2018-07-10 17:10:37 -07:00
|
|
|
indented_plain_text (Text):
|
2018-07-12 16:14:29 -07:00
|
|
|
{~ (("\\" -> "\") / (("\" nl_nodent "..") -> "") / (!text_interpolation "\") / [^%nl\]+)+
|
|
|
|
(%nl+ (nodent -> ""))* ~}
|
2018-01-30 15:13:04 -08:00
|
|
|
text_interpolation:
|
2018-07-12 16:14:29 -07:00
|
|
|
inline_text_interpolation / ("\" indented_expression nl_nodent "..")
|
2017-12-30 14:31:07 -08:00
|
|
|
|
2018-07-12 16:14:29 -07:00
|
|
|
number (Number): (("-"? (([0-9]+ "." [0-9]+) / ("." [0-9]+) / ([0-9]+)))-> tonumber)
|
2017-12-30 14:31:07 -08:00
|
|
|
|
|
|
|
-- Variables can be nameless (i.e. just %) and can't contain operators like apostrophe
|
|
|
|
-- which is a hack to allow %'s to parse as "%" and "' s" separately
|
2018-07-12 16:14:29 -07:00
|
|
|
variable (Var): "%" {(%ident_char+ ((!"'" %operator_char+) / %ident_char+)*)?}
|
2017-12-30 14:31:07 -08:00
|
|
|
|
|
|
|
inline_list (List):
|
2018-01-19 17:29:44 -08:00
|
|
|
!('[..]')
|
2018-05-30 13:07:08 -07:00
|
|
|
"[" %ws*
|
2018-07-12 16:14:29 -07:00
|
|
|
(inline_list_item (%ws* ',' %ws* inline_list_item)* (%ws* ',')?)? %ws*
|
2018-05-30 13:07:08 -07:00
|
|
|
("]" / (","? (
|
2018-07-12 16:14:29 -07:00
|
|
|
(!! eol -> "Line ended before finding a closing ]-bracket" !!)
|
|
|
|
/(!! [^%nl]* -> "Unexpected character while parsing List" !!)
|
2018-05-30 13:07:08 -07:00
|
|
|
)))
|
2017-12-30 14:31:07 -08:00
|
|
|
indented_list (List):
|
2018-07-12 16:14:29 -07:00
|
|
|
"[..]" nl_indent
|
|
|
|
list_line (nl_nodent list_line)*
|
|
|
|
(&nl_dedent / (","? (!! (!nl_dedent .)* -> "Unexpected character while parsing List" !!)))
|
2017-12-30 14:31:07 -08:00
|
|
|
list_line:
|
2018-07-12 16:14:29 -07:00
|
|
|
(inline_list_item %ws* "," %ws*)+ eol
|
|
|
|
/ (inline_list_item %ws* "," %ws*)* (action / expression)
|
2018-04-27 16:45:11 -07:00
|
|
|
inline_list_item: inline_block / inline_action / inline_expression
|
2017-12-30 14:31:07 -08:00
|
|
|
|
2018-01-03 00:52:01 -08:00
|
|
|
inline_dict (Dict):
|
2018-01-19 17:29:44 -08:00
|
|
|
!('{..}')
|
2018-05-30 13:07:08 -07:00
|
|
|
"{" %ws*
|
2018-07-12 16:14:29 -07:00
|
|
|
(inline_dict_entry (%ws* ',' %ws* inline_dict_entry)*)? %ws*
|
2018-05-30 13:07:08 -07:00
|
|
|
("}" / (","? (
|
2018-07-12 16:14:29 -07:00
|
|
|
(!! %ws* eol -> "Line ended before finding a closing }-brace" !!)
|
|
|
|
/ (!! [^%nl]* -> "Unexpected character while parsing Dictionary" !!)
|
2018-05-30 13:07:08 -07:00
|
|
|
)))
|
2018-01-03 00:52:01 -08:00
|
|
|
indented_dict (Dict):
|
2018-07-12 16:14:29 -07:00
|
|
|
"{..}" nl_indent
|
|
|
|
dict_line (nl_nodent dict_line)*
|
|
|
|
(&nl_dedent / (","? (!! (!nl_dedent .)* -> "Unexpected character while parsing Dictionary" !!)))
|
2018-01-03 00:52:01 -08:00
|
|
|
dict_line:
|
2018-07-12 16:14:29 -07:00
|
|
|
(inline_dict_entry %ws* "," %ws*)+ eol
|
|
|
|
/ (inline_dict_entry %ws* "," %ws*)* dict_entry
|
2018-05-16 15:44:07 -07:00
|
|
|
dict_entry(DictEntry):
|
2018-07-12 16:14:29 -07:00
|
|
|
dict_key (%ws* ":" %ws* (action / expression))?
|
2018-05-16 15:44:07 -07:00
|
|
|
inline_dict_entry(DictEntry):
|
2018-07-12 16:14:29 -07:00
|
|
|
dict_key (%ws* ":" %ws* (inline_block / inline_action / inline_expression)?)?
|
2018-04-11 20:05:12 -07:00
|
|
|
dict_key:
|
2018-05-10 22:47:03 -07:00
|
|
|
text_word / inline_expression
|