diff options
author | Daniel Friesel <derf@finalrewind.org> | 2017-05-16 15:48:34 +0200 |
---|---|---|
committer | Daniel Friesel <derf@finalrewind.org> | 2017-05-16 15:48:34 +0200 |
commit | 458cf889e6ed331d9e5084cf8e199166ce983286 (patch) | |
tree | 11f070baffdb5589599d47985655881dc115681c /lib/Kratos | |
parent | 2531355bd6c7020904e889164efde3a5bb19631f (diff) |
autogenerate simple models (without args or parameters) from AC++ attributes
Diffstat (limited to 'lib/Kratos')
-rw-r--r-- | lib/Kratos/DFADriver.pm | 17 | ||||
-rw-r--r-- | lib/Kratos/DFADriver/Model.pm | 81 |
2 files changed, 95 insertions, 3 deletions
diff --git a/lib/Kratos/DFADriver.pm b/lib/Kratos/DFADriver.pm index 8c8f780..7ceaa4a 100644 --- a/lib/Kratos/DFADriver.pm +++ b/lib/Kratos/DFADriver.pm @@ -35,11 +35,24 @@ sub new { $self->{dfa} = Kratos::DFADriver::DFA->new(%opt); $self->{mimosa} = MIMOSA->new(%opt); - $self->{model} = Kratos::DFADriver::Model->new(%opt); $self->{repo} = AspectC::Repo->new; - $self->{class_name} = $self->{model}->class_name; $self->{lp}{iteration} = 1; + if ( -r $opt{xml_file} ) { + $self->{model} = Kratos::DFADriver::Model->new(%opt); + $self->{class_name} = $self->{model}->class_name; + } + elsif ( $opt{class_name} ) { + $self->{model} = Kratos::DFADriver::Model->new_from_repo( + repo => $self->{repo}, + class_name => $opt{class_name}, + xml_file => $opt{xml_file}, + ); + } + else { + die('Neither driver.xml nor class name specified, cannot continue'); + } + bless( $self, $class ); $self->set_paths; diff --git a/lib/Kratos/DFADriver/Model.pm b/lib/Kratos/DFADriver/Model.pm index 677fd99..f87b81c 100644 --- a/lib/Kratos/DFADriver/Model.pm +++ b/lib/Kratos/DFADriver/Model.pm @@ -8,7 +8,7 @@ use parent 'Class::Accessor'; use Carp; use Carp::Assert::More; -use List::Util qw(first); +use List::Util qw(first uniq); use XML::LibXML; Kratos::DFADriver::Model->mk_ro_accessors(qw(class_name xml)); @@ -32,6 +32,85 @@ sub new { return $self; } +sub new_from_repo { + my ( $class, %opt ) = @_; + my $repo = $opt{repo}; + my $class_name = $opt{class_name}; + + my @states; + my %transition; + + if ( not exists $repo->{class}{$class_name} ) { + die("Unknown class: $class_name\n"); + } + my $class_base = $repo->{class}{$class_name}; + + for my $function ( values %{ $class_base->{function} } ) { + for my $attrib ( @{ $function->{attributes} // [] } ) { + if ( $attrib =~ s{ ^ src _ }{}x ) { + push( @states, $attrib ); + push( @{ $transition{ $function->{name} }{src} }, $attrib ); + } + elsif ( $attrib =~ s{ ^ dst _ }{}x ) { + push( @states, $attrib ); + push( @{ $transition{ $function->{name} }{dst} }, $attrib ); + } + else { + say "wat $attrib"; + } + } + } + + @states = uniq @states; + @states = sort @states; + + my $xml = XML::LibXML::Document->new('1.0'); + + my $data_node = $xml->createElement('data'); + my $driver_node = $xml->createElement('driver'); + my $states_node = $xml->createElement('states'); + my $transitions_node = $xml->createElement('transitions'); + + $driver_node->setAttribute( name => $class_name ); + + $xml->setDocumentElement($data_node); + $data_node->appendChild($driver_node); + $driver_node->appendChild($states_node); + $driver_node->appendChild($transitions_node); + + for my $state (@states) { + my $state_node = $xml->createElement('state'); + $state_node->setAttribute( name => $state ); + $states_node->appendChild($state_node); + } + for my $transition ( sort keys %transition ) { + my $transition_node = $xml->createElement('transition'); + my $level = $transition eq 'epilogue' ? 'epilogue' : 'user'; + $transition_node->setAttribute( name => $transition ); + for my $src_state ( @{ $transition{$transition}{src} } ) { + my $src_node = $xml->createElement('src'); + my $src_name = $xml->createTextNode($src_state); + $src_node->appendChild($src_name); + $transition_node->appendChild($src_node); + } + for my $dst_state ( @{ $transition{$transition}{dst} } ) { + my $dst_node = $xml->createElement('dst'); + my $dst_name = $xml->createTextNode($dst_state); + $dst_node->appendChild($dst_name); + $transition_node->appendChild($dst_node); + } + my $level_node = $xml->createElement('level'); + my $level_name = $xml->createTextNode($level); + $level_node->appendChild($level_name); + $transition_node->appendChild($level_node); + $transitions_node->appendChild($transition_node); + } + + $xml->toFile( $opt{xml_file} ); + + return $class->new(%opt); +} + sub parse_xml_property { my ( $self, $node, $property_name ) = @_; |