/examples/json.nqp

http://github.com/plobsing/nqp · Unknown · 106 lines · 85 code · 21 blank · 0 comment · 0 complexity · 1c70e0605b920351d90199b79c6b40a1 MD5 · raw file

  1. #! nqp
  2. # A JSON compiler written in NQP. To use this compiler, first
  3. # precompile the code to PIR, then run that:
  4. #
  5. # $ nqp --target=pir json.nqp >json.pir
  6. # $ parrot json.pir
  7. #
  8. # It can then be turned into a .pbc to be available as load_language:
  9. #
  10. # $ parrot -o json.pbc json.pir
  11. # $ cp json.pbc <installroot>/lib/<version>/languages
  12. #
  13. INIT {
  14. pir::load_bytecode('P6Regex.pbc');
  15. pir::load_bytecode('dumper.pbc');
  16. }
  17. grammar JSON::Grammar is HLL::Grammar {
  18. rule TOP { <value> }
  19. proto token value { <...> }
  20. token value:sym<string> { <string> }
  21. token value:sym<number> {
  22. '-'?
  23. [ <[1..9]> <[0..9]>+ | <[0..9]> ]
  24. [ '.' <[0..9]>+ ]?
  25. [ <[Ee]> <[+\-]>? <[0..9]>+ ]?
  26. }
  27. rule value:sym<array> {
  28. '[' [ <value> ** ',' ]? ']'
  29. }
  30. rule value:sym<object> {
  31. '{'
  32. [ [ <string> ':' <value> ] ** ',' ]?
  33. '}'
  34. }
  35. token string {
  36. <?["]> <quote_EXPR: ':qq'>
  37. }
  38. }
  39. class JSON::Actions is HLL::Actions {
  40. method TOP($/) {
  41. make PAST::Block.new($<value>.ast, :node($/));
  42. };
  43. method value:sym<string>($/) { make $<string>.ast; }
  44. method value:sym<number>($/) { make +$/; }
  45. method value:sym<array>($/) {
  46. my $past := PAST::Op.new(:pasttype<list>, :node($/));
  47. if $<value> {
  48. for $<value> { $past.push($_.ast); }
  49. }
  50. make $past;
  51. }
  52. method value:sym<object>($/) {
  53. my $past := PAST::Stmts.new( :node($/) );
  54. my $hashname := PAST::Compiler.unique('hash');
  55. my $hash := PAST::Var.new( :scope<register>, :name($hashname),
  56. :viviself('Hash'), :isdecl );
  57. my $hashreg := PAST::Var.new( :scope<register>, :name($hashname) );
  58. $past.'push'($hash);
  59. # loop through all string/value pairs, add set opcodes for each pair.
  60. my $n := 0;
  61. while $n < +$<string> {
  62. $past.'push'(PAST::Op.new( :pirop<set__vQ~*>, $hashreg,
  63. $<string>[$n].ast, $<value>[$n].ast ) );
  64. $n++;
  65. }
  66. # return the Hash as the result of this node
  67. $past.'push'($hashreg);
  68. make $past;
  69. }
  70. method string($/) { make $<quote_EXPR>.ast; }
  71. }
  72. class JSON::Compiler is HLL::Compiler {
  73. INIT {
  74. JSON::Compiler.language('json');
  75. JSON::Compiler.parsegrammar(JSON::Grammar);
  76. JSON::Compiler.parseactions(JSON::Actions);
  77. }
  78. method autoprint($value) {
  79. _dumper($value, 'JSON')
  80. unless (pir::getinterp__P()).stdhandle(1).tell > $*AUTOPRINTPOS;
  81. }
  82. our sub MAIN(@ARGS) is pirflags<:main> {
  83. JSON::Compiler.command_line(@ARGS);
  84. }
  85. }