handle descent into complex data
[scpubgit/JSON-Tree-Viewer.git] / br.pl
diff --git a/br.pl b/br.pl
index 6ac30cc..613687b 100644 (file)
--- a/br.pl
+++ b/br.pl
@@ -14,7 +14,7 @@ sub _build_root {
 }
 
 sub _build_json {
-  JSON->new->relaxed
+  JSON->new->relaxed->pretty
 }
 
 sub dispatch_request {
@@ -24,49 +24,88 @@ sub dispatch_request {
       ref($_[0][0]) eq 'HASH' ? $self->render_table($_[0][0]) : $_[0]
     }
   },
+  sub (/) {
+    [ $self->structure($self->root) ];
+  },
   sub (/**/) {
-    $self->structure($self->descend($self->root, split '/', $_[1]));
+    [ $self->structure($self->descend($self->root, split '/', $_[1])) ];
   },
 }
 
 sub structure {
   my ($self, $data) = @_;
+  return unless $data;
   if (ref($data) eq 'HASH') {
-    $data = [ @{$data}{sort keys %$data} ];
-    my @cols = sort keys %{$data->[0]};
-    return [ {
-      columns => \@cols,
-      data => $data,
-    } ];
+    if (keys %$data > 1
+        and values %$data == grep ref($_) eq 'HASH', values %$data) {
+      my %tmp;
+      $tmp{join '|', keys %$_} = 1 for values %$data;
+      if (keys %tmp == 1) {
+        $data->{$_}->{name} ||= $_ for keys %$data;
+        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 [{
+      columns => [ 'key', 'value' ],
+      data => [ map +{ key => $_, value => $data->{$_} }, sort keys %$data ],
+    }];
+  } elsif (blessed($data) and $data->isa('IO::All::Dir')) {
+    return [{
+      columns => [ 'name', 'explore' ],
+      data => [
+        map +{ name => $_, explore => $self->link_to($_) }, keys %$data,
+      ]
+    }];
   } else {
     die "Confused by $data";
   }
 }
 
+sub link_to {
+  my ($self, @to) = @_;
+  use HTML::Tags;
+  s/\//\./g for my @link = @to;
+  my $link = join('/', @link, '');
+  my $to = $to[-1];
+  my $html = join '', HTML::Tags::to_html_string(
+    <a href="${link}">, "Explore $to", </a>
+  );
+  return \$html;
+}
+
 sub descend {
   my ($self, $target, @path) = @_;
-  return $target unless @path;
-  my $step = shift @path;
+  return unless $target;
   if (blessed($target) and $target->isa('IO::All::File')) {
     $target = $self->json->decode(scalar $target->all);
   }
-  $self->descend($target->{$step}, @path);
+  return $target unless @path;
+  (my $undot = my $step = shift @path) =~ s/\./\//g;
+  $self->descend($target->{$step}||$target->{$undot}, @path);
 }
 
 sub render_table {
   my ($self, $data) = @_;
   use HTML::Tags;
   my @rows = (
-    $data->{columns},
-    map [ @{$_}{@{$data->{columns}}} ], @{$data->{data}}
+    $data->{show_columns} ? { map +($_ => $_), @{$data->{columns}} } : (),
+    @{$data->{data}}
   );
   [ 200, [ 'Content-type' => 'text/html' ], [
     HTML::Tags::to_html_string(
       <html>, <body>, "\n",
       <table>, "\n",
-        (map {;
+        (map { my $el = $_;
           '  ', <tr>,
-            (map { <td>, $self->render_el($_), </td> } @$_),
+            (map {
+              <td>, $self->render_el($el, $_), </td>
+            } @{$el}{@{$data->{columns}}}),
           </tr>, "\n"
         } @rows),
       </table>, "\n",
@@ -75,6 +114,21 @@ sub render_table {
   ] ];
 }
 
-sub render_el { ref($_[1]) eq 'ARRAY' ? join(', ', @{$_[1]}) : $_[1] }
+sub render_el {
+  my ($self, $whole, $key, $part) = @_;
+  if (ref($part) eq 'ARRAY') {
+    return join(', ', @$part);
+  }
+  if (ref($part) eq 'HASH') {
+    if ($whole->{key}) {
+      return $self->link_to($whole->{key})
+    } elsif ($whole->{name}) {
+      return $self->link_to($whole->{name}, $key);
+    }
+    $part = '(complex)';
+  }
+  use HTML::Tags;
+  return $part =~ /\n/ ? (<pre>, $part, </pre>) : $part;
+}
 
 __PACKAGE__->run_if_script;