Beefy Boxes and Bandwidth Generously Provided by pair Networks
Syntactic Confectionery Delight
 
PerlMonks  

Re^2: Parsing a macro language

by simonm (Vicar)
on Jul 09, 2005 at 01:20 UTC ( [id://473626]=note: print w/replies, xml ) Need Help??


in reply to Re: Parsing a macro language
in thread Parsing a macro language

Or, for a hash-based represenation that's closer to what you originally asked for:

sub parse_brackets { my @parse; my @stack = \@parse; my $fewer_indents = 1; # Try setting this to 0 or 1 my $line_no; foreach my $line ( @_ ) { $line_no ++; $line =~ s/\A\s+//; $line =~ s/\s+\Z//; if ( $line !~ /\S/ ) { next; } elsif ( $line =~ s/\s*\{$// ) { my @line = split ' ', $line; my %node = ( type => shift(@line), ( @line ? ( name => shift(@li +ne) ) : () ) ); push @{ $stack[0] }, \%node; unshift @stack, do { $node{contents} = [] }; } elsif ( $line eq '}' ) { shift @stack; scalar @stack or die("Too many right brackets at line $line_no") +; } else { push @{ $stack[0] }, $line; } } scalar @stack == 1 or die("Too few right brackets at line $line_no") +; return @parse; }

Output:

$VAR1 = { 'contents' => [ { 'contents' => [ { 'contents' => [ 'Do you +like your pie with ice cream?' ], 'type' => 'label' }, { 'contents' => [ '1 Yes', '2 No' ], 'type' => 'single' } ], 'name' => '4B', 'type' => 'question' }, { 'contents' => [ { 'contents' => [ 'Do you +like your pie with whipped cream?' ], 'type' => 'label' }, { 'contents' => [ '1 Yes', '2 No' ], 'type' => 'single' } ], 'name' => '4C', 'type' => 'question' } ], 'name' => 'p1', 'type' => 'page' };

Oh, and for future reference, I don't think I would call this a "macro language". It's a data markup format, with no functional or substitutional behavior.

Replies are listed 'Best First'.
Re^3: Parsing a macro language
by simonm (Vicar) on Jul 09, 2005 at 03:31 UTC
    You could decouple the parsing and building steps, as nothingmuch suggested:
    sub parse_brackets { my @out; foreach my $line ( @_ ) { $line_no ++; $line =~ s/\A\s+//; $line =~ s/\s+\Z//; if ( $line !~ /\S/ ) { next; } elsif ( $line =~ s/\s*\{$// ) { push @out, 'open' => [ split ' ', $line ]; } elsif ( $line eq '}' ) { push @out, 'close' => 1; } else { push @out, 'line' => $line; } } return @out; } sub build_brackets { my @parse; my @stack = \@parse; while ( scalar @_ ) { my ( $type, $value ) = splice( @_, 0, 2 ); if ( $type eq 'open' ) { my %node = ( type => shift(@$value), ( @$value ? ( name => shift +(@$value) ) : () ) ); push @{ $stack[0] }, \%node; unshift @stack, do { $node{contents} = [] }; } elsif ( $type eq 'close' ) { shift @stack; scalar @stack or die("Too many right brackets"); } else { push @{ $stack[0] }, $value; } } scalar @stack == 1 or die("Too few right brackets"); return @parse; } use Data::Dumper; print Dumper( build_brackets( parse_brackets( @lines ) ) );

    I'm not sure this is a big win for your case, but if you need the extra flexiblity...

Log In?
Username:
Password:

What's my password?
Create A New User
Domain Nodelet?
Node Status?
node history
Node Type: note [id://473626]
help
Chatterbox?
and the web crawler heard nothing...

How do I use this?Last hourOther CB clients
Other Users?
Others romping around the Monastery: (2)
As of 2024-04-19 21:15 GMT
Sections?
Information?
Find Nodes?
Leftovers?
    Voting Booth?

    No recent polls found