Group
Extension

Chart-Plotly/tool/generate_from_schema.pl

#!/usr/bin/env perl 

use strict;
use warnings;
use utf8;

use Getopt::Long::Descriptive;
use Const::Fast;
use Path::Tiny;
use Text::Template;
use JSON;
use Cwd;
use Data::Dump;
use Scalar::Util;

# TODO update
# TODO Use enum names
# TODO Use enums with JSON::false and JSON::true and number
# TODO Types: color, subplotid, angle, colorscale
# TODO Add defaults?
# TODO Add support for items


my ($opt, $usage) = describe_options($0 . '%o',
   ['plotlyjs_version|p=s', 'Plotly.js version to generate from', {required => 1}],
   ['temp_path|t=s', 'Temporary path to checkout plotly.js and switch to plotlyjs version', {default => 'plotlyjsrepository'}],
   [],
   ['help', 'Display help and exit']
);

print($usage->text), exit if $opt->help;


my $plotlyjs_base_path = path($opt->temp_path);
if (!$plotlyjs_base_path->exists) {
    system("git clone https://github.com/plotly/plotly.js " . $plotlyjs_base_path);
} else {
    system("cd $plotlyjs_base_path && git checkout master && git pull origin");
}
# TODO check that this went well
system("cd $plotlyjs_base_path && git checkout " . $opt->plotlyjs_version);

my $moose_type_for = {
    any        => 'Any',
    number     => 'Num',
    string     => 'Str',
    flaglist   => 'Str',
    color      => 'Str',
    boolean    => 'Bool',
    integer    => 'Int',
    info_array => 'ArrayRef|PDL',
    data_array => 'ArrayRef|PDL',
    enumerated => => 'enum'
};
my $template = path("template/trace.tmpl")->slurp_utf8();
my $attribute_template = path("template/attribute.tmpl")->slurp_utf8();
my $plotly_js_dist_path = $plotlyjs_base_path->child("dist");
my $current_dir = cwd;
my $types_without_moose_equivalent = {};

my $plotly_schema = from_json($plotly_js_dist_path->child('plot-schema.json')->slurp_utf8());
my $traces_schema = $plotly_schema->{'traces'};

for my $trace_name (sort keys %$traces_schema) {
    my $trace_schema = $traces_schema->{$trace_name};
    my $ast = GenerateTraceAST($trace_schema, $trace_name);
    RenderTypeAST($trace_name, $ast, $template, $trace_name);
}

print "Types without Moose equivalent: \n" . join("\n", sort keys %$types_without_moose_equivalent) . "\n";

$plotly_js_dist_path->child('plotly.min.js')->copy(path($current_dir)->child('share')->child('plotly.js'));
$plotlyjs_base_path->child('LICENSE')->copy(path($current_dir)->child('share')->child('plotly.js'));
my $changes_file = path('Changes');
my @changes = $changes_file->lines;
if (!grep /\Q$opt->plotlyjs_version\E/, @changes) {
    unshift @changes, '  - Update to plotly.js ' . $opt->plotlyjs_version . "\n";
    $changes_file->spew(@changes);
}

my $plotlypm_file = path('lib')->child('Chart')->child('Plotly.pm');
$plotlypm_file->edit_lines(sub {
        if (/\s+return '([0-9.]+)'; # plotlyjs_version_tag$/) {
            my $version = $opt->plotlyjs_version;
            $version =~ s/v//;
            $_ = "    return '$version'; # plotlyjs_version_tag\n";
        }
    });



sub FieldsAST {
    my $fields_schema = shift();
    my $parent_class = shift();
    my $AST = shift();
    for my $field_name (sort keys %$fields_schema) {
        if ($field_name eq "_deprecated") {
            next;
        }
        if ($field_name eq "role") {
            next;
        }
        if ($field_name eq "editType") {
            next;
        }
        if ($field_name eq "arrayOk") {
            next;
        }
        if ($field_name eq "dflt") {
            next;
        }
        if ($field_name eq "type") {
            next;
        }
        my $field_contents = $fields_schema->{$field_name};
        if (ref $field_contents eq 'HASH') {
            if (exists $field_contents->{'role'} && $field_contents->{'role'} eq "object") {
                if (exists $field_contents->{items}) {
                    if (ref $field_contents->{items} eq 'HASH' && scalar keys %{$field_contents->{items}} == 1) {
                        my ($item_name) = keys %{$field_contents->{items}};
                        $AST->{subtypes}{$item_name} = SubtypeAST($field_contents->{items}{$item_name}, $item_name, $parent_class);
                        my $field = {
                            is  => 'rw',
                            isa => Data::Dump::quote("ArrayRef|ArrayRef[" . GenerateClassName($parent_class, $item_name) . "]")
                        };
                        $AST->{fields}{$field_name} = $field;
                    } else {
                        warn("Role object with items with more than 1 type of item. Ignored");
                    }
                } else {
                    $AST->{subtypes}{$field_name} = SubtypeAST($field_contents, $field_name, $parent_class);
                    my $field = {
                        is  => 'rw',
                        isa => Data::Dump::quote("Maybe[HashRef]|" . GenerateClassName($parent_class, $field_name))
                    };
                    if (defined $field_contents->{arrayOk} && $field_contents->{arrayOk}) {
                        warn("Until now this combination is not present (array of elements with role object). Ignored");
                    }
                    $AST->{fields}{$field_name} = $field;
                }
            }
            else {
                my $field = {
                    is => 'rw'
                };
                if (defined $field_contents->{'description'}) {
                    $field->{documentation} = $field_contents->{'description'};
                }
                if (defined $field_contents->{'valType'}) {
                    my $moose_type = $moose_type_for->{$field_contents->{'valType'}};
                    if (defined $moose_type) {
                        if ($moose_type eq 'enum') {
                            if (defined $field_contents->{values}) {
                                my $only_strings = 1;
                                for my $value (@{$field_contents->{values}}) {
                                    if (Scalar::Util::looks_like_number($value)) {
                                        $only_strings = 0;
                                    }
                                }

                                if ($only_strings) {
                                    my $enum_type = 'enum([' . join(",", map {Data::Dump::quote($_)} @{$field_contents->{values}}) . '])';
                                    if (defined $field_contents->{arrayOk} && $field_contents->{arrayOk}) {
                                        $field->{isa} = "union([" . $enum_type . ", " .  Data::Dump::quote("ArrayRef") . "])";
                                    } else {
                                        $field->{isa} = $enum_type;
                                    }
                                }
                            }
                        }
                        else {
                            if (defined $field_contents->{arrayOk} && $field_contents->{arrayOk}) {
                                $field->{isa} = Data::Dump::quote($moose_type . "|ArrayRef[" . $moose_type . "]");
                            } else {
                                $field->{isa} = Data::Dump::quote($moose_type);
                            }
                        }
                    }
                    else {
                        $types_without_moose_equivalent->{$field_contents->{'valType'}} = 1;
                    }
                }
                if (defined $field_contents->{arrayOk} && $field_contents->{arrayOk} && !defined $field->{isa}) {
                    $field->{isa} = Data::Dump::quote("Maybe[ArrayRef]");
                }

                $AST->{fields}{$field_name} = $field;
            }

        }
        else {
            $AST->{fields}{$field_name} = {
                default => $field_contents,
                is      => 'ro'
            };
        }
    }
}

sub GenerateClassName {
    my $parent_class = shift();
    my $type_name = shift();
    return $parent_class . '::' . ucfirst($type_name);
}

sub InitialAST {
    my $class_name = shift();
    return {
        class_name => $class_name,
        fields     => {},
        subtypes   => {}
    };
}

sub SubtypeAST {
    my $type_schema = shift();
    my $type_name = shift();
    my $parent_class = shift();

    my $class_name = GenerateClassName($parent_class, $type_name);
    my $AST = InitialAST($class_name);
    FieldsAST($type_schema, $class_name, $AST);
    return $AST;
}

sub GenerateTraceAST {
    my $trace_schema = shift();
    my $trace_name = shift();

    my $class_name = GenerateClassName('Chart::Plotly::Trace', $trace_name);
    my $AST = InitialAST($class_name);

    if (defined $trace_schema->{'meta'}{'description'}) {
        $AST->{documentation} = $trace_schema->{'meta'}{'description'};
    }

    my $fields_schema = $trace_schema->{attributes};
    FieldsAST($fields_schema, $class_name, $AST);
    return $AST;
}


sub RenderField {
    my $field_name = shift();
    if ($field_name eq "meta") { # Collision with Moose meta attribute
        $field_name = 'pmeta'; # Plotly meta
    } 
    my $ast = shift();

    my $file_contents = "=item * " . $field_name . "\n";
    my $documentation;
    if (defined $ast->{'documentation'}) {
        $documentation = $ast->{'documentation'};
        $documentation =~ s/M<(.+?)>/$1/g;
        $file_contents .= "\n" . $documentation;
    }
    $file_contents .= "\n\n=cut\n\n";
    $file_contents .= "has $field_name => (\n    is => " . Data::Dump::quote($ast->{is}) . ",";
    if (defined $ast->{isa}) {
        $file_contents .= "\n    isa => " . $ast->{isa} . ",";
    }
    if (defined $ast->{default}) {
        $file_contents .= "\n    default => " . Data::Dump::quote($ast->{default}) . ",";
    }
    if (defined $documentation) {
        $file_contents .= "\n    documentation => " . Data::Dump::quote($documentation) . ",";
    }
    return $file_contents .= "\n);\n\n";
}

sub RenderTypeAST {
    my $trace_name = shift();
    my $ast = shift();
    my $template = shift();
    my $root_trace_name = shift();

    my $file_contents = "";

    for my $field (sort keys %{$ast->{fields}}) {
        my $value = $ast->{fields}{$field};
        $file_contents .= RenderField($field, $value);
    }
    $file_contents .= "=pod\n\n=back\n\n=cut\n\n";
    $file_contents .= "\n__PACKAGE__->meta->make_immutable();\n";
    $file_contents .= "1;\n";

    my $used_modules = "";
    for my $subtype (sort keys %{$ast->{subtypes}}) {
        RenderTypeAST($subtype, $ast->{subtypes}{$subtype}, $attribute_template, $root_trace_name);
        my $type_constraint = $ast->{subtypes}{$subtype}{class_name};
        $used_modules .= "use $type_constraint;\n";
    }

    my $description = $ast->{documentation};
    my $header =
        Text::Template::fill_in_string($template, HASH => {
            package_name => $ast->{class_name},
            trace_name   => $root_trace_name,
            used_modules => $used_modules,
            description  => $description
        });

    $file_contents = $header . $file_contents;
    chdir $current_dir;
    my $file = path('lib/' . join("/", split(/::/, $ast->{class_name})) . ".pm");
    $file->parent->mkpath();
    $file->spew_utf8($file_contents);
}


Powered by Groonga
Maintained by Kenichi Ishigaki <ishigaki@cpan.org>. If you find anything, submit it on GitHub.