🎄teratailクリスマスプレゼントキャンペーン2024🎄』開催中!

\teratail特別グッズやAmazonギフトカード最大2,000円分が当たる!/

詳細はこちら
Ruby

Rubyはプログラミング言語のひとつで、オープンソース、オブジェクト指向のプログラミング開発に対応しています。

Q&A

解決済

1回答

604閲覧

Parsletライブラリを使用したrubyでのparserにてエラーが起こる

light180

総合スコア161

Ruby

Rubyはプログラミング言語のひとつで、オープンソース、オブジェクト指向のプログラミング開発に対応しています。

0グッド

0クリップ

投稿2019/12/13 05:01

編集2019/12/14 04:53

Parsletライブラリを使用してPEGを記述しparserを作っていたのですが、Parslet::Parsefaildというraiseエラーが起きてしまいます。
どなたか、ご教示いただけると、嬉しいです。

追記、解決しました。@asmさん、ありがとうございます。

Error

1Traceback (most recent call last): 2 2: from parser.rb:66:in `<main>' 3 1: from /Library/Ruby/Gems/2.6.0/gems/parslet-1.8.2/lib/parslet/atoms/base.rb:49:in `parse' 4/Library/Ruby/Gems/2.6.0/gems/parslet-1.8.2/lib/parslet/cause.rb:70:in `raise': Expected at least 1 of LINE ' (Parslet::ParseFailed) 5' at line 1 char 1.

以下、新しいエラーです。

Traceback (most recent call last): 2: from temp.rb:64:in `<main>' 1: from /Library/Ruby/Gems/2.6.0/gems/parslet-1.8.2/lib/parslet/atoms/base.rb:49:in `parse' /Library/Ruby/Gems/2.6.0/gems/parslet-1.8.2/lib/parslet/cause.rb:70:in `raise': Extra input after last repetition at line 2 char 1. (Parslet::ParseFailed)

parser.rb

1require 'parslet' 2class NumericParser < Parslet::Parser 3 idens = ["print"] 4 root(:code) 5 rule(:space){ str(" ") } 6 rule(:spaces){ space.repeat(1) } 7 rule(:space?){ spaces.maybe } 8 rule(:return_mark){ str("\n") } 9 rule(:returns){ return_mark.repeat(1) } 10 rule(:return?){ returns.maybe } 11 rule(:sprt?){ (return_mark | space).repeat(0)} 12 rule(:chars){ str("a") | str("b") | str("c") | str("d") | str("e") | str("f") | str("g") | str("h") | str("i") | str("j") | str("k") | str("l") | str("m") | str("n") | str("o") | str("p") | str("q") | str("r") | str("s") | str("t") | str("u") | str("v") | str("w") | str("x") | str("y") | str("z") | str("A") | str("B") | str("C") | str("D") | str("E") | str("F") | str("G") | str("H") | str("I") | str("J") | str("K") | str("L") | str("M") | str("N") | str("O") | str("P") | str("Q") | str("R") | str("S") | str("T") | str("U") | str("V") | str("W") | str("X") | str("Y") | str("Z") | str("0") | str("1") | str("2") | str("3") | str("4") | str("5") | str("6") | str("7") | str("8") | str("9") | str(" ") | str("!") | str("\\"") | str("#") | str("$") | str("%") | str("&") | str("\'") | str("(") | str(")") | str("-") | str("^") | str("@") | str("[") | str(";") | str(":") | str("]") | str(",") | str(".") | str("/") | str("\\") | str("=") | str("~") | str("|") | str("`") | str("{") | str("+") | str("*") | str("}") | str("<") | str(">") | str("?") | str("_") | str("\n") | str("\s") | str("\t") } 13 14 rule(:string) { 15 str("\"") >> chars.repeat >> str("\"") 16 } 17 18 rule(:var) { 19 match("[a-z]") >> match("[a-zA-Z1234567890]").repeat 20 } 21 22 rule(:integer) { 23 match("[0-9]").repeat(1) 24 } 25 26 rule(:code) { 27 (line >> str("\n")).repeat(1) 28 } 29 30 rule(:line) { 31 func | assign | value | sprt? 32 } 33 34 rule(:func) { 35 idens.map{|f| str(f)}.inject(:|) >> block 36 } 37 38 rule(:block){ 39 block_value | block_lists 40 } 41 42 rule(:block_value){ 43 str("(") >> sprt? >> line >> sprt? >> str(")") 44 } 45 46 rule(:block_lists){ 47 str("{") >> sprt? >> code >> sprt? >>str("}") 48 } 49 50 rule(:value){ 51 string | var | integer | func 52 } 53 54 rule(:assign) { 55 var >> space? >> str("=") >> space? >> value 56 } 57end 58f = open(ARGV[0]) 59code = [] 60while line = f.gets 61 code.push(line) 62end 63f.close 64NumericParser.new.parse(code.join("\n")+"\n")

テストデータです。

123
print(123)
print("aiueo")
print{ print("aiueo") }

気になる質問をクリップする

クリップした質問は、後からいつでもMYページで確認できます。

またクリップした質問に回答があった際、通知やメールを受け取ることができます。

バッドをするには、ログインかつ

こちらの条件を満たす必要があります。

guest

回答1

0

ベストアンサー

ruby

1 root(:code) 2 rule(:code) { 3 (line >> str("\n")).repeat(1) 4 } 5 6 rule(:line) { 7 func | assign | value 8 }

空行で死にませんか?


  • rule(:chars)にstr("")が混じっているとrepeatした時無限にメモリを食い潰します。
  • matchは1文字以上にマッチさせる事はできません。

まずは、小さなテストからやってみるべきです。

ruby

1np = NumericParser.new 2np.integer.parse %(123) 3np.string.parse %("aiueo") 4np.func.parse %(print(123)) 5np.func.parse %(print("aiueo")) 6np.parse <<CODE 7print{ 8print("aiueo") 9} 10CODE

これで、どこが間違ったルールなのか分かるかと思います。

funcだけはちょっと面倒なので答えておきます。

ruby

1 rule(:func) { 2 idens.map{|f| str(f)}.inject(:|) >> block 3 }

投稿2019/12/13 05:27

編集2019/12/14 04:45
asm

総合スコア15149

バッドをするには、ログインかつ

こちらの条件を満たす必要があります。

light180

2019/12/13 05:32

> 空白で死ぬ どう言うことですか?
asm

2019/12/13 05:41

現象が再現できるテストデータがないと確実な事は言えませんが 空白行がテストデータに含まれた場合、このルールではマッチしない気がします。 code.join("\n")が期待通りの文字列かを検証し、ご提示願います。
light180

2019/12/13 05:44

なるほど。更新しました。お願いします。 ``` require 'parslet' class NumericParser < Parslet::Parser idens = ["print"] root(:code) rule(:space){ str(" ") } rule(:spaces){ space.repeat(1) } rule(:space?){ spaces.maybe } rule(:return_mark){ str("\n") } rule(:returns){ return_mark.repeat(1) } rule(:return?){ returns.maybe } rule(:sprt?){ (return_mark | space).repeat(0)} rule(:chars){ str("!") | str("#") | str("$") | str("%") | str("&") | str("") | str("(") | str(")") | str("*") | str("+") | str(",") | str("-") | str(".") | str("/") | str("0") | match("[0-9]") | str(":") | str(";") | str("<") | str("=") | str(">") | str("?") | str("@") | match("[A-Z]") | str("[") | str("\") | str("]") | str("^") | str("_") | str("`") | match("[a-z]") | str("{") | str("|") | str("}") | str("~") | str("\\"") } rule(:string) { str("\"") >> chars.repeat >> str("\"") } rule(:var) { match("[a-z]") >> match("[a-zA-Z1234567890]").repeat } rule(:integer) { match("[0-9]") } rule(:code) { (line >> str("\n")).repeat(1) } rule(:line) { func | assign | value | sprt? } rule(:func) { match("#{idens.join("|")}") >> block } rule(:block){ block_value | block_lists } rule(:block_value){ str("(") >> sprt? >> line >> sprt? >> str(")") } rule(:block_lists){ str("{") >> sprt? >> code >> sprt? >>str("}") } rule(:value){ string | var | integer | func } rule(:assign) { var >> space? >> str("=") >> space? >> value } end f = open(ARGV[0]) code = [] while line = f.gets code.push(line) end f.close NumericParser.new.parse(code.join("\n")) ```
light180

2019/12/13 05:45

本文の方も、修正しておきますね。
asm

2019/12/13 06:24 編集

あ、わかりました。 code.join("\n")によって、最終行の改行が消えるのが問題です。 code.join("\n")+"\n"のように補うと良さげ 追記: 他にも問題ありそう。 とりあえず、テストデータの提示をお願いします。
light180

2019/12/14 03:52

返信、ありがとうございます。直してみましたが、エラーは変わらず、出てしまいます。 テストデータなど、アップロードしました。(追記しました)
light180

2019/12/14 04:55 編集

ありがとうございます!!そのように、一部を指定してparseできるんですね。デバックに役立ちそうです。また、parsletの|をinjectで使えることに驚きました。お陰さまで、小さなテストではエラーが出ないようになりましたが、file指定したときに、エラーが出てしまいます。アドバイス頂けると、嬉しいです。
asm

2019/12/14 09:46

ファイル読み込み部分が一行ずつ読み込んでいるのには理由があるのでしょうか? code = open(ARGV[0], &:read) NumericParser.new.parse(code) でいいと思うのですが 理由あっての場合は while line = f.gets code.push(line.chomp) end にした方が素直なように思います。
light180

2019/12/27 11:20

お返事、遅くなりすみません。 まだ、同じエラーが出てしまうのですが、お返事が遅れたこともありますので、ベストアンサーに選ばさせていただきます。 このエラーは別の質問で出そうと思います。 下にリンクを貼ったので、見ていただけたら嬉しいです。
guest

あなたの回答

tips

太字

斜体

打ち消し線

見出し

引用テキストの挿入

コードの挿入

リンクの挿入

リストの挿入

番号リストの挿入

表の挿入

水平線の挿入

プレビュー

15分調べてもわからないことは
teratailで質問しよう!

ただいまの回答率
85.36%

質問をまとめることで
思考を整理して素早く解決

テンプレート機能で
簡単に質問をまとめる

質問する

関連した質問