have no rows marker work as expected with columns
[scpubgit/JSON-Tree-Viewer.git] / br.pl
diff --git a/br.pl b/br.pl
index 0bb29b6..08fe3b6 100644 (file)
--- a/br.pl
+++ b/br.pl
 package TB_Temp_Packname;
 
 use Web::Simple;
+use Module::Runtime qw(use_module);
 use Scalar::Util qw(blessed);
 use IO::All;
 use JSON;
+use URI::Escape;
 
 has root => (is => 'lazy');
 
 has json => (is => 'lazy');
 
 sub _build_root {
-  io->dir("/home/matthewt/tmp/introspection-data/host/services-dev/stable/node/host/")
+  io->dir( $ENV{JTV_ROOT} || die "JTV_ROOT env var not set" )
 }
 
 sub _build_json {
-  JSON->new->relaxed
+  JSON->new->relaxed->pretty
 }
 
 sub dispatch_request {
   my ($self) = @_;
+  sub (/raw/...) {
+    use_module('Plack::App::Directory')->new({
+      root => $self->root->name
+    });
+  },
   sub () {
     response_filter {
       ref($_[0][0]) eq 'HASH' ? $self->render_table($_[0][0]) : $_[0]
     }
   },
   sub (/) {
-    [ $self->structure($self->root) ];
+    sub (?@host=) {
+      [ 302, [ 'Location', '/'.join('+', @{$_[1]}).'/' ], [] ];
+    },
+    sub () {
+      $self->root_structure;
+    },
   },
   sub (/**/) {
-    [ $self->structure($self->descend($self->root, split '/', $_[1])) ];
+    $self->structure(map {
+      s{\\/}{/}g;
+      s{\\\\}{\\}g;
+      $_;
+    } split qr{(?<!\\)/}, $_[1]);
   },
 }
 
+sub root_structure {
+  my ($self) = @_;
+  my $struct = $self->mangle_structure($self->root);
+  push @{$struct->[0]{columns}}, 'select';
+  foreach my $host (@{$struct->[0]{data}}) {
+    use HTML::Tags;
+    my $name = $host->{name};
+    my $html = join '', HTML::Tags::to_html_string(
+      <input type="checkbox" name="host" value="$name" />
+    );
+    $host->{select} = \$html;
+  }
+  $struct->[0]{wrapper} = sub {
+    use HTML::Tags;
+    '  ', <form>, "\n",
+    (map /^\s*$/ ? "    $_" : $_, @_),
+    '    ', <input type="submit" value="Now Multify" />, "\n",
+    '  ', </form>, "\n"
+  };
+  return $struct;
+}
+
 sub structure {
+  my ($self, @parts) = @_;
+  if ($parts[0] =~ /\+/) {
+    ($parts[0], my @extra) = split /\+/, $parts[0];
+    my $struct = $self->mangle_structure($self->descend($self->root, @parts));
+    my $first = shift @parts;
+    my @rest = map [ $_, $self->mangle_structure(
+                     $self->descend($self->root, $_, @parts)
+                   )->[0] ], @extra;
+    unless ($struct->[0]{show_columns}) {
+      my @cols = @{ $struct->[0]{columns} };
+      if (@cols == 2) {
+        my ($key_name, $value_name) = @cols;
+        my %name;
+        $name{ $_ }++
+          for map $_->{$key_name},
+              map @$_, $struct->[0]{data}, map $_->[1]{data}, @rest;
+        my %value_by_host = (map {
+          my $host = $_->[0];
+          my $data = $_->[1]{data};
+          ($host, +{
+            map { ($_->{$key_name}, $_->{$value_name}) } @$data,
+          });
+        } [$first, $struct->[0]], @rest);
+        my @hosts = ($first, @extra);
+        return [{
+          columns => ['key', @hosts],
+          show_columns => 1,
+          data => [ map {
+            my $key = $_;
+            +{ key => $key, (map {
+              ($_, $value_by_host{$_}{$key});
+            } @hosts)};
+          } sort keys %name ],
+        }];
+      }
+    }
+    my %by_name;
+    my %host_cols;
+    my %complex_cols;
+    foreach my $thing ([ $first, $struct->[0] ], @rest) {
+      foreach my $el (@{$thing->[1]{data}}) {
+        my $by = $by_name{$el->{name}} ||= { name => $el->{name} };
+        foreach my $key (keys %$el) {
+          next if $key eq 'name';
+          if (ref($el->{$key}) eq 'HASH') {
+            $complex_cols{$key} = 1;
+            $by->{$key} = {};
+          } else {
+            my $full_key = $key.' ('.$thing->[0].')';
+            $host_cols{$full_key} = 1;
+            $by->{$full_key} = $el->{$key};
+          }
+        }
+      }
+    }
+    return [{
+      columns => [ 'name', sort(keys %host_cols), sort(keys %complex_cols) ],
+      show_columns => 1,
+      data => [
+        map $by_name{$_}, sort keys %by_name
+      ],
+    }];
+  }
+  return $self->mangle_structure($self->descend($self->root, @parts));
+}
+
+sub mangle_structure {
   my ($self, $data) = @_;
+  return unless $data;
   if (ref($data) eq 'HASH') {
     if (keys %$data > 1
         and values %$data == grep ref($_) eq 'HASH', values %$data) {
@@ -41,42 +147,65 @@ sub structure {
       $tmp{join '|', keys %$_} = 1 for values %$data;
       if (keys %tmp == 1) {
         $data->{$_}->{name} ||= $_ for keys %$data;
-        return {
-          columns => [ sort keys %{(values %$data)[0]} ],
+        my @cols = grep $_ ne 'name', sort keys %{(values %$data)[0]};
+        unshift @cols, 'name';
+        return [{
+          columns => \@cols,
           show_columns => 1,
           data => [ @{$data}{sort keys %$data} ],
-        }
+        }]
       }
     }
-    return {
+    return [{
       columns => [ 'key', 'value' ],
       data => [ map +{ key => $_, value => $data->{$_} }, sort keys %$data ],
-    };
+    }];
+  } elsif (ref($data) eq 'ARRAY') {
+    if (not grep { not ref($_) eq 'HASH' } @$data) {
+      my %key;
+      $key{$_} = 1
+        for map { keys %$_ } @$data;
+      return [{
+        columns => [sort keys %key],
+        show_columns => 1,
+        data => $data,
+      }];
+    }
   } elsif (blessed($data) and $data->isa('IO::All::Dir')) {
-    return {
+    return [{
       columns => [ 'name', 'explore' ],
       data => [
         map +{ name => $_, explore => $self->link_to($_) }, keys %$data,
       ]
-    };
+    }];
   } else {
     die "Confused by $data";
   }
 }
 
 sub link_to {
-  my ($self, $to) = @_;
+  my ($self, @to) = @_;
   use HTML::Tags;
+  my @link = map {
+    my $link = $_;
+    $link =~ s{\\}{\\\\}g;
+    $link =~ s{/}{\\/}g;
+    $link;
+  } @to;
+  my $link = join('/', @link, '');
+  my $to = $to[-1];
   my $html = join '', HTML::Tags::to_html_string(
-    <a href="${to}/">, "Explore $to", </a>
+    <a href="./${link}">, "Explore $to", </a>
   );
   return \$html;
 }
 
 sub descend {
   my ($self, $target, @path) = @_;
+  return unless $target;
   if (blessed($target) and $target->isa('IO::All::File')) {
-    $target = $self->json->decode(scalar $target->all);
+    my $all = $target->all;
+    $target = $self->json->decode($all);
   }
   return $target unless @path;
   my $step = shift @path;
@@ -90,32 +219,58 @@ sub render_table {
     $data->{show_columns} ? { map +($_ => $_), @{$data->{columns}} } : (),
     @{$data->{data}}
   );
+  my $column_count = scalar @{$data->{columns}};
   [ 200, [ 'Content-type' => 'text/html' ], [
     HTML::Tags::to_html_string(
       <html>, <body>, "\n",
-      <table>, "\n",
-        (map { my $el = $_;
-          '  ', <tr>,
-            (map {
-              <td>, $self->render_el($el, $_), </td>
-            } @{$el}{@{$data->{columns}}}),
-          </tr>, "\n"
-        } @rows),
-      </table>, "\n",
+      ($data->{wrapper}||sub{@_})->(
+        '', <table>, "\n",
+          (map { my $el = $_;
+            '  ', ($el->{key} eq '__error__') ? <tr class="error"> : <tr>,
+              (map {
+                <td>, $self->render_el($el, $_, $el->{$_}), </td>
+              } @{$data->{columns}}),
+            </tr>, "\n"
+          } @rows),
+          @{$data->{data}}
+          ? ()
+          : (<tr class="no-rows">,
+              <td colspan="$column_count">,
+                'No entries in this data structure',
+              </td>,
+            </tr>),
+        '', </table>, "\n",
+      ),
       </body>, </html>, "\n",
     )
   ] ];
 }
 
 sub render_el {
-  my ($self, $whole, $part) = @_;
+  my ($self, $whole, $key, $part) = @_;
   if (ref($part) eq 'ARRAY') {
-    return join(', ', @$part);
+    if (grep { ref($_) eq 'HASH' } @$part) {
+      if ($whole->{key}) {
+        return $self->link_to($whole->{key})
+      } elsif ($whole->{name}) {
+        return $self->link_to($whole->{name}, $key);
+      }
+    }
+    return join ', ', @$part
+      if @$part < 5;
+    use HTML::Tags;
+    return <ul>, (map { (<li>, $_, </li>) } @$part), </ul>;
   }
   if (ref($part) eq 'HASH') {
-    return $self->link_to($whole->{key});
+    if ($whole->{key}) {
+      return $self->link_to($whole->{key})
+    } elsif ($whole->{name}) {
+      return $self->link_to($whole->{name}, $key);
+    }
+    $part = '(complex)';
   }
-  return $part;
+  use HTML::Tags;
+  return $part =~ /\n/ ? (<pre>, $part, </pre>) : $part;
 }
 
 __PACKAGE__->run_if_script;