bitbake: toaster: image information views
authorDave Lerner <dave.lerner@windriver.com>
Wed, 12 Mar 2014 21:54:09 +0000 (16:54 -0500)
committerRichard Purdie <richard.purdie@linuxfoundation.org>
Fri, 21 Mar 2014 14:47:53 +0000 (14:47 +0000)
[YOCTO # 4346]

When a target image is selected, this commit adds to the toaster
project a two-tabbed page that shows
1) 'packages included' a table of packages included in the image
(see target.html), and
2) 'directory structure', the target image's file system directory
and detailed information showing the source of each file in the
directory table (see dirinfo.html).

The directory structure tab relies on the open source jQuery plugin
jtreetable which provides hierarchical table expansions and contractions
of the directory entry tables as the user drills down into directories.

A file of jtreetable styles that are compatible with other toaster styles
is provided included as css/jquery.treetable.theme.toaster.css. The
complete unaltered jtreetable plugin is added via a separate commit.

This work was developed base on the bugzilla specification number 4346
and the document "Design 1.1 Image information" attached to that report.

Whitespace and typo fixes from Alex Damian.

(Bitbake rev: 1ba9f310a8b4fd0952a95be86ab43ae27fe6d983)

Signed-off-by: Dave Lerner <dave.lerner@windriver.com>
Signed-off-by: Alexandru DAMIAN <alexandru.damian@intel.com>
Signed-off-by: Richard Purdie <richard.purdie@linuxfoundation.org>
12 files changed:
bitbake/lib/toaster/orm/models.py
bitbake/lib/toaster/toastergui/static/css/default.css
bitbake/lib/toaster/toastergui/static/css/jquery.treetable.theme.toaster.css [new file with mode: 0644]
bitbake/lib/toaster/toastergui/static/jquery.treetable.theme.toaster.css [new file with mode: 0644]
bitbake/lib/toaster/toastergui/templates/base.html
bitbake/lib/toaster/toastergui/templates/basebuildpage.html
bitbake/lib/toaster/toastergui/templates/dirinfo.html [new file with mode: 0644]
bitbake/lib/toaster/toastergui/templates/package_included_detail.html
bitbake/lib/toaster/toastergui/templates/target.html
bitbake/lib/toaster/toastergui/templatetags/projecttags.py
bitbake/lib/toaster/toastergui/urls.py
bitbake/lib/toaster/toastergui/views.py

index 93506d7..c5fe69b 100644 (file)
@@ -58,6 +58,9 @@ class Target(models.Model):
     image_size = models.IntegerField(default=0)
     license_manifest_path = models.CharField(max_length=500, null=True)
 
+    def package_count(self):
+        return Target_Installed_Package.objects.filter(target_id__exact=self.id).count()
+
     def __str__(self):
         return self.target
 
@@ -194,7 +197,7 @@ class Task_Dependency(models.Model):
     depends_on = models.ForeignKey(Task, related_name='task_dependencies_depends')
 
 class Package(models.Model):
-    search_allowed_fields = ['name', 'version', 'revision', 'recipe__name', 'recipe__version', 'recipe__license', 'recipe__layer_version__layer__name', 'recipe__layer_version__branch', 'recipe__layer_version__commit', 'recipe__layer_version__layer__local_path']
+    search_allowed_fields = ['name', 'version', 'revision', 'recipe__name', 'recipe__version', 'recipe__license', 'recipe__layer_version__layer__name', 'recipe__layer_version__branch', 'recipe__layer_version__commit', 'recipe__layer_version__layer__local_path', 'installed_name']
     build = models.ForeignKey('Build')
     recipe = models.ForeignKey('Recipe', null=True)
     name = models.CharField(max_length=100)
index 7db156a..53a3fee 100644 (file)
@@ -104,7 +104,7 @@ select { width: auto; }
 .well > .lead, .alert .lead { margin-bottom: 0px; }
 .no-results { margin: 10px 0; }
 .task-name { margin-left: 7px; }
-
+.icon-hand-right {color: #ccccc; }
 
 
 
diff --git a/bitbake/lib/toaster/toastergui/static/css/jquery.treetable.theme.toaster.css b/bitbake/lib/toaster/toastergui/static/css/jquery.treetable.theme.toaster.css
new file mode 100644 (file)
index 0000000..d8552e5
--- /dev/null
@@ -0,0 +1,38 @@
+table.treetable span.file {
+  background-image: url();
+}
+
+table.treetable span.folder {
+  background-image: url();
+}
+
+table.treetable tr.collapsed span.indenter a {
+  background-image: url();
+}
+
+table.treetable tr.expanded span.indenter a {
+  background-image: url();
+}
+
+
+
+table.treetable tr.collapsed.selected span.indenter a {
+  background-image: url();
+}
+
+table.treetable tr.expanded.selected span.indenter a {
+  background-image: url();
+}
+
+table.treetable tr.accept {
+  background-color: #a3bce4;
+  color: #fff
+}
+
+table.treetable tr.collapsed.accept td span.indenter a {
+  background-image: url();
+}
+
+table.treetable tr.expanded.accept td span.indenter a {
+  background-image: url();
+}
diff --git a/bitbake/lib/toaster/toastergui/static/jquery.treetable.theme.toaster.css b/bitbake/lib/toaster/toastergui/static/jquery.treetable.theme.toaster.css
new file mode 100644 (file)
index 0000000..5194b23
--- /dev/null
@@ -0,0 +1,66 @@
+/*
+table.treetable {
+  border: 1px solid #888; 
+  border-collapse: collapse;
+  font-size: .8em;
+  line-height: 1;
+  margin: .6em 0 1.8em 0;
+  width: 100%;
+}
+
+table.treetable caption {
+  font-size: .9em;
+  font-weight: bold;
+  margin-bottom: .2em;
+}
+
+table.treetable tbody tr td {
+  cursor: default;
+  padding: .3em 1em;
+}
+
+table.treetable span {
+  background-position: center left;
+  background-repeat: no-repeat;
+  padding: .2em 0 .2em 1.5em;
+}
+*/
+
+table.treetable span.file {
+  background-image: url();
+}
+
+table.treetable span.folder {
+  background-image: url();
+}
+
+table.treetable tr.collapsed span.indenter a {
+  background-image: url();
+}
+
+table.treetable tr.expanded span.indenter a {
+  background-image: url();
+}
+
+
+
+table.treetable tr.collapsed.selected span.indenter a {
+  background-image: url();
+}
+
+table.treetable tr.expanded.selected span.indenter a {
+  background-image: url();
+}
+
+table.treetable tr.accept {
+  background-color: #a3bce4;
+  color: #fff
+}
+
+table.treetable tr.collapsed.accept td span.indenter a {
+  background-image: url();
+}
+
+table.treetable tr.expanded.accept td span.indenter a {
+  background-image: url();
+}
index 5493e23..9ca9c9a 100644 (file)
@@ -49,6 +49,8 @@ function reload_params(params) {
 }
 </script>
 
+{% block extraheadcontent %}
+{% endblock %}
     </head>
 
 <body style="height: 100%">
index 054a37c..636fca2 100644 (file)
@@ -26,7 +26,7 @@
         <div id="nav" class="span2">
             <ul class="nav nav-list well">
                 <li class="nav-header">Images</li>
-            {% for t in build.target_set.all %}
+            {% for t in build.target_set.all|dictsort:"target" %}
                 <li><a href="{% url 'target' build.pk t.pk %}">{{t.target}}</a><li>
             {% endfor %}
                 <li class="nav-header">Build</li>
diff --git a/bitbake/lib/toaster/toastergui/templates/dirinfo.html b/bitbake/lib/toaster/toastergui/templates/dirinfo.html
new file mode 100644 (file)
index 0000000..9b76a1c
--- /dev/null
@@ -0,0 +1,237 @@
+{% extends "basebuildpage.html" %}
+{% block extraheadcontent %}
+{% load static %}
+<link rel="stylesheet" href="{% static 'css/jquery.treetable.css' %}" type="text/css">
+<link rel="stylesheet" href="{% static 'css/jquery.treetable.theme.toaster.css' %}" type="text/css">
+{% endblock extraheadcontent %}
+
+{% block localbreadcrumb %}
+<li>{{target.target}}</li>
+{% endblock localbreadcrumb%}
+
+{% block buildinfomain %}
+
+{% load static %}
+<script src="{% static 'js/jquery.treetable.js' %}">
+</script>
+{% load projecttags %}
+
+<script type='text/javascript'>
+    function setupTreetable() {
+        $("#dirtable").treetable({
+            expandable: true,
+            branchAttr: "ttBranch",
+            clickableNodeNames: true,
+            onNodeCollapse: function() {
+                /* Do nothing, keep cached */
+            },
+            onNodeExpand: function() {
+                var start = this.id;
+                var n = $("#dirtable").treetable("node", start);
+                if (this.children.length > 0) {
+                    /* already was expanded once */
+                    $("#dirtable").treetable("reveal", start);
+                }
+                else {
+                    var url = "{% url "dirinfo_ajax" build.id target.id  %}";
+                    $.ajax({
+                        async: false,
+                        type    : "GET",
+                        url     : url,
+                        data    : "start=" + start,
+                        success : function(response) {
+                            var objects = $.parseJSON(response);
+                            addRows(n, objects)
+                        },
+                        error   : function(jqXHR, textStatus, errorThrown ) {alert(textStatus + ":" + errorThrown)},
+                     });
+                }
+             },
+       });
+    }
+    function td(data) {
+        if (data == null) {
+            data = '';
+        }
+        return '<td>' + data + '</td>'
+    }
+
+    function formatRow(o) {
+        /* setup tr-wide formatting */
+        var tr = '<tr class="';
+        if (o.link_to != null) {
+            tr += 'muted ';
+        }
+        if (o.isdir && o.childcount) {
+            tr += 'branch" data-tt-branch="true" ';
+        }
+        else {
+            tr += 'leaf" data-tt-branch="false" ';
+        }
+        tr +=    ' data-tt-id="' + o.fullpath +'" ';
+        if (o.parent != "/") {
+            tr +=    ' data-tt-parent-id="' + o.parent +'" ';
+        }
+        tr += '>';
+
+        /* setup td specific formatting */
+        var link_to = td(o.link_to);
+        var size = td(o.size);
+        var permission = td(o.permission);
+        var owner = td(o.owner);
+        var group = td(o.group);
+
+        /* handle the name column */
+        var name = null;;
+        var namespan=1;
+        if (o.isdir) {
+            if (o.link_to == null) {
+                namespan = 2;
+                if (o.package == null) {
+                    namespan = 3;
+                }
+            }
+            var colspan = 'colspan="' + namespan + '"';
+            name = '<td class="content-directory"' + colspan + '>';
+            if (o.childcount) {
+                name += '<a href="">';
+            }
+            name += '<i class="icon-folder-close"></i>';
+            name += '&nbsp;' + o.name;
+            if (o.childcount) {
+                name += '</a>';
+            }
+            name += '</td>';
+        }
+        else {
+            name = '<td>';
+            if (o.link_to == null) {
+                name += '<i class="icon-file"></i>';
+            }
+            else {
+                name += '<i class="icon-hand-right"></i>';
+            }
+            name += '&nbsp;' + o.name;
+            name += '</td>';
+        }
+
+        /* handle the package column */
+        var package = null;
+        if (o.package != null) {
+            /* add link to included package page */
+            build_id = {{ build.id }};
+            target_id = {{ target.id }};
+            /* Create a url for a dummy package id of 0 */
+            dummy = "{% url 'package_included_detail' build.id target.id 0 %}"
+            /* fill in the package id */
+            url = dummy.substr(0, dummy.length-1) + o.package_id;
+            package = '<a href=' + url + '>' ;
+            package += o.package;
+            package += '</a>';
+            if (o.installed_package != o.package) {
+                /* make class muted and add hover help */
+                package += '<span class="muted"> as ' + o.installed_package + ' </span>';
+                package += '<i class="icon-question-sign get-help hover-help" ';
+                package += 'title="' + o.package + ' was renamed at packaging time and was installed in your image as ' + o.installed_package + '">';
+                package += '</i>';
+            }
+        }
+        package = td(package);
+
+        var cols1to3;
+        switch (namespan) {
+            case 3:
+                cols1to3 = name;
+                break;
+            case  2:
+                cols1to3 = name + package;
+                break;
+            default:
+                cols1to3 = name + link_to + package;
+        }
+        r = tr + cols1to3 + size + permission + owner + group + "</tr>"
+        return r;
+    }
+
+    function addRows(n, objs) {
+        rows = "";
+        for (i=0; i<objs.length; i++) {
+            rows += formatRow(objs[i]);
+        }
+        $("#dirtable").treetable("loadBranch", n, rows);
+    }
+
+    $.fn.isOffScreen = function(){
+        var win = $(window);
+        viewportBottom = win.scrollTop() + win.height();
+        
+        var bounds = this.offset();
+        bounds.bottom = bounds.top + this.outerHeight();
+        
+        return (bounds.bottom > viewportBottom);
+    };
+
+    function selectRow(path) {
+        var row  = $('tr[data-tt-id="' + path + '"]');
+        row.addClass(" highlight");
+        if (row.isOffScreen()) {
+            $('html, body').animate({ scrollTop: row.offset().top - 150}, 2000);
+        }
+    }
+</script>
+
+<div class="span10">
+
+    <div class="page-header">
+        <h1> {{target.target}} </h1>
+    </div>
+
+    <ul class="nav nav-pills">
+        <li class="">
+            <a href="{% url 'target' build.id target.id %}">
+            <i class="icon-question-sign get-help" data-toggle="tooltip" title="Of all the packages built, the subset installed in the root file system of this image"></i>
+                Packages included ({{target.package_count}} - {{packages_sum|filtered_filesizeformat}})
+            </a>
+        </li>
+        <li class="active">
+            <a href="{% url 'dirinfo' build.id target.id %}">
+                <i class="icon-question-sign get-help" data-toggle="tooltip" title="The directories and files in the root file system of this image"></i>
+                Directory structure
+            </a>
+        </li>
+    </ul>
+
+    <div id="directory-structure" class="tab-pane active">
+        <table id="dirtable" class="table table-bordered table-hover treetable">
+            <thead>
+                <tr>
+                    <th>Directory / File</th>
+                    <th>Symbolic link to</th>
+                    <th>Source package</th>
+                    <th>Size</th>
+                    <th>Permissions</th>
+                    <th>Owner</th>
+                    <th>Group</th>
+                </tr>
+            </thead>
+            <tbody>
+            <script type='text/javascript'>
+                setupTreetable();
+                addRows(null, {{ objects|safe }} );
+                {% if file_path %}
+                    {% comment %}
+                        link from package_included_detail specifies file path
+                    {% endcomment %}
+                    {% for dir_elem in dir_list %}
+                        $("#dirtable").treetable("expandNode", "{{dir_elem}}");
+                    {% endfor %}
+                    selectRow("{{file_path}}");
+                {% endif %}
+            </script>
+            </tbody>
+        </table>
+    </div> <!-- directory-structure -->
+</div> <!-- span10 -->
+
+{% endblock buildinfomain %}
+
index df25885..ce4f1cb 100644 (file)
@@ -24,7 +24,7 @@
                                        {% for file in package.buildfilelist_package.all|dictsort:"path" %}
                         <tr>
                             <td>
-                                <a href="{% url 'image_information_dir' build.id target.id file.id %}">
+                                <a href="{% url 'dirinfo_filepath' build.id target.id file.path %}">
                                     {{file.path}}
                                 </a>
                              </td>
index f2d0ad4..4512898 100644 (file)
@@ -1,8 +1,153 @@
 {% extends "basebuildpage.html" %}
-
 {% block localbreadcrumb %}
-<li>Target</li>
-{% endblock %}
+<li>{{target.target}}</li>
+{% endblock localbreadcrumb%}
+
+{% load projecttags %}
 
 {% block buildinfomain %}
-{% endblock %}
+
+<div class="row-fluid span10">
+    <div class="page-header">
+        <h1>
+            {% if request.GET.search and objects.paginator.count > 0 %}
+                {{objects.paginator.count}} package{{objects.paginator.count|pluralize}} found
+            {% elif request.GET.search and objects.paginator.count == 0 %}
+                No packages found
+            {% else %}
+                {{target.target}}
+            {% endif %}
+        </h1>
+    </div>
+</div>
+
+<div class="row-fluid pull-right span10" id="navTab">
+    <ul class="nav nav-pills">
+        <li class="active">
+            <a href="#target">
+            <i class="icon-question-sign get-help" data-toggle="tooltip" title="Of all the packages built, the subset installed in the root file system of this image"></i>
+                Packages included ({{target.package_count}} - {{packages_sum|filtered_filesizeformat}})
+            </a>
+        </li>
+        <li>
+            <a href="{% url 'dirinfo' build.id target.id %}">
+                <i class="icon-question-sign get-help" data-toggle="tooltip" title="The directories and files in the root file system of this image"></i>
+                Directory structure
+            </a>
+        </li>
+    </ul>
+
+    <div id="image-packages" class="tab-pane">
+
+    {% if objects.paginator.count == 0 %}
+    <div class="row-fluid">
+        <div class="alert">
+            <form class="no-results input-append" id="searchform">
+                <input id="search" name="search" class="input-xxlarge" type="text" value="{{request.GET.search}}"/>{% if request.GET.search %}<a href="javascript:$('#search').val('');searchform.submit()" class="add-on btn" tabindex="-1"><i class="icon-remove"></i></a>{% endif %}
+                <button class="btn" type="submit" value="Search">Search</button>
+                <button class="btn btn-link" onclick="javascript:$('#search').val('');searchform.submit()">Show all packages</button>
+            </form>
+        </div>
+    </div>
+
+
+    {% else %}
+    {% include "basetable_top.html" %}
+    {% for package in objects %}
+    <tr>
+        <td class="package_name">
+            <a href="{% url 'package_included_detail' build.id target.id package.id %}">
+            {{package.name}}
+            </a>
+            {% if package.installed_name and package.name != package.installed_name %}
+            <span class="muted"> as {{package.installed_name}}</span>
+            <i class="icon-question-sign get-help hover-help" title='{{package.name|add:" was renamed at packaging time and was installed in your image as "|add:package.installed_name}}'></i>
+            {% endif %}
+        </td>
+        <td class="package_version">
+            <a href="{% url 'package_included_detail' build.id target.id package.id %}">
+            {{package.version|filtered_packageversion:package.revision}}
+            </a>
+        </td>
+        <td class="package_size">
+            {{package.size|filtered_installedsize:package.installed_size|filtered_filesizeformat}}
+        </td>
+        <td class="size_over_total">
+            {{package|filter_sizeovertotal:packages_sum}}
+        </td>
+        <td class="license">
+            {{package.license}}
+        </td>
+        <td class="depends">
+            {% with deps=package|runtime_dependencies:target.id %}
+            {% with deps_count=deps|length %}
+            {% if deps_count > 0 %}
+            <a class="btn"
+                title="<a href='{% url "package_included_dependencies" build.id target.id package.id %}'>{{package.name}}</a> depends on"
+                data-content="<ul class='unstyled'>
+                  {% for i in deps|dictsort:'depends_on.name' %}
+                    <li><a href='{% url "package_included_dependencies" build.pk target.id i.depends_on.pk %}'>{{i.depends_on.name}}</a></li>
+                  {% endfor %}
+                </ul>">
+                {{deps_count}}
+            </a>
+            {% endif %}
+            {% endwith %}
+            {% endwith %}
+        </td>
+        <td class="brought_in_by">
+            {% with rdeps=package|reverse_runtime_dependencies:target.id %}
+            {% with rdeps_count=rdeps|length %}
+            {% if rdeps_count > 0 %}
+            <a class="btn"
+                title="<a href='{% url "package_included_reverse_dependencies" build.id target.id package.id %}'>{{package.name}}</a> is brought in by"
+                data-content="<ul class='unstyled'>
+                  {% for i in rdeps|dictsort:'package.name' %}
+                    <li><a href='{% url "package_included_dependencies" build.id target.id i.package.id %}'>{{i.package.name}}</a></li>
+                  {% endfor %}
+                </ul>">
+                {{rdeps_count}}
+            </a>
+            {% endif %}
+            {% endwith %}
+            {% endwith %}
+        </td>
+        <td class="recipe_name">
+            {% if package.recipe.version %}
+            <a href="{% url 'recipe' build.id package.recipe_id %}">
+            {{ package.recipe.name }}
+            </a>
+            {% endif %}
+        </td>
+        <td class="recipe_version">
+            {% if package.recipe.version %}
+            <a href="{% url 'recipe' build.id package.recipe_id %}">
+            {{ package.recipe.version }}
+            </a>
+            {% endif %}
+        </td>
+        <td class="layer_name">
+            {{ package.recipe.layer_version.layer.name }}
+        </td>
+        <td class="layer_branch">
+            {{ package.recipe.layer_version.branch}}
+        </td>
+        <td class="layer_commit">
+            <a class="btn"
+                data-content="<ul class='unstyled'>
+                  <li>{{package.recipe.layer_version.commit}}</li>
+                </ul>">
+                {{package.recipe.layer_version.commit|truncatechars:13}}
+            </a>
+        </td>
+        <td class="layer_directory">
+            {{ package.recipe.layer_version.layer.local_path }}
+        </td>
+    </tr>
+    {% endfor %}
+
+    {% include "basetable_bottom.html" %}
+    {% endif %}
+    </div> <!-- tabpane -->
+</div> <!--span 10-->
+{% endblock buildinfomain %}
index 60d5dd0..e08258b 100644 (file)
@@ -215,3 +215,52 @@ def get_image_extensions( build ):
             comma = ", ";
     return( extensions );
 
+@register.filter
+def filtered_installedsize(size, installed_size):
+    """If package.installed_size not null and not empty return it,
+       else return package.size
+    """
+    return size if (installed_size == 0) or (installed_size == "") or (installed_size == None) else installed_size
+
+@register.filter
+def filtered_installedname(name, installed_name):
+    """If package.installed_name not null and not empty
+        return <div class=muted> as {{package.installed_name}}
+        otherwise ""
+    """
+    return name if (name == installed_name) or (not installed_name) or (installed_name == "") else name + " as " + installed_name
+
+@register.filter
+def filtered_packageversion(version, revision):
+    """ Emit "version-revision" if version and revision are not null
+        else "version" if version is not null
+        else ""
+    """
+    return "" if (not version or version == "") else version if (not revision or revision == "") else version + "-" + revision
+        
+from django.db import models
+from orm.models import Package
+@register.filter
+def runtime_dependencies(package_object, targetid):
+    """ Return a queryset that lists the packages this package depends on
+    """
+    return package_object.package_dependencies_source.filter(target_id__exact=targetid, dep_type__in={'1'})
+
+@register.filter
+def reverse_runtime_dependencies(package_object, targetid):
+    """ Return a queryset that lists the packages depending on this package
+    """
+    return package_object.package_dependencies_target.filter(target_id__exact = targetid,dep_type__in={'1'})
+
+@register.filter
+def filter_sizeovertotal(package_object, total_size):
+    """ Return the % size of the package over the total size argument
+        formatted nicely.
+    """
+    size = package_object.installed_size
+    if size == None or size == '':
+        size = package_object.size
+  
+    return '{:.1%}'.format(float(size)/float(total_size))
+
+
index 8be27b0..ac83b38 100644 (file)
@@ -45,8 +45,10 @@ urlpatterns = patterns('toastergui.views',
 
         # images are known as targets in the internal model
         url(r'^build/(?P<build_id>\d+)/target/(?P<target_id>\d+)$', 'target', name='target'),
+        url(r'^dentries/build/(?P<build_id>\d+)/target/(?P<target_id>\d+)$', 'dirinfo_ajax', name='dirinfo_ajax'),
+        url(r'^build/(?P<build_id>\d+)/target/(?P<target_id>\d+)/dirinfo$', 'dirinfo', name='dirinfo'),
+        url(r'^build/(?P<build_id>\d+)/target/(?P<target_id>\d+)/dirinfo_filepath/(?P<file_path>(?:/[^/\n]+)*)$', 'dirinfo', name='dirinfo_filepath'),
         url(r'^build/(?P<build_id>\d+)/target/(?P<target_id>\d+)/packages$', 'tpackage', name='targetpackages'),
-
         url(r'^build/(?P<build_id>\d+)/configuration$', 'configuration', name='configuration'),
         url(r'^build/(?P<build_id>\d+)/configvars$', 'configvars', name='configvars'),
         url(r'^build/(?P<build_id>\d+)/buildtime$', 'buildtime', name='buildtime'),
index 9740ef3..97514cc 100644 (file)
 
 import operator
 
-from django.db.models import Q
+from django.db.models import Q, Sum
 from django.shortcuts import render, redirect
 from orm.models import Build, Target, Task, Layer, Layer_Version, Recipe, LogMessage, Variable
 from orm.models import Task_Dependency, Recipe_Dependency, Package, Package_File, Package_Dependency
-from orm.models import Target_Installed_Package
+from orm.models import Target_Installed_Package, Target_File
 from django.views.decorators.cache import cache_control
 from django.core.paginator import Paginator, EmptyPage, PageNotAnInteger
 from django.http import HttpResponseBadRequest
 from django.utils import timezone
 from datetime import timedelta
 from django.utils import formats
+import json
 
 def _build_page_range(paginator, index = 1):
     try:
@@ -163,7 +164,7 @@ def _get_search_results(search_term, queryset, model):
 def _search_tuple(request, model):
     ordering_string, invalid = _validate_input(request.GET.get('orderby', ''), model)
     if invalid:
-        raise BaseException("Invalid ordering " + str(invalid))
+        raise BaseException("Invalid ordering model:" + str(model) + str(invalid))
 
     filter_string, invalid = _validate_input(request.GET.get('filter', ''), model)
     if invalid:
@@ -284,8 +285,8 @@ def builds(request):
                  'qhelp': "The date and time the build finished",
                  'orderfield': _get_toggle_order(request, "completed_on", True),
                  'ordericon':_get_toggle_order_icon(request, "completed_on"),
-                 'filter' : {'class' : 'completed_on', 
-                             'label': 'Show:', 
+                 'filter' : {'class' : 'completed_on',
+                             'label': 'Show:',
                              'options' : [
                                          ("Today's builds", 'completed_on__gte:'+timezone.now().strftime("%Y-%m-%d"), queryset_with_search.filter(completed_on__gte=timezone.now().strftime("%Y-%m-%d")).count()),
                                          ("Yesterday's builds", 'completed_on__gte:'+(timezone.now()-timedelta(hours=24)).strftime("%Y-%m-%d"), queryset_with_search.filter(completed_on__gte=(timezone.now()-timedelta(hours=24)).strftime("%Y-%m-%d")).count()),
@@ -307,8 +308,8 @@ def builds(request):
                  'qhelp': "How many errors were encountered during the build (if any)",
                  'orderfield': _get_toggle_order(request, "errors_no", True),
                  'ordericon':_get_toggle_order_icon(request, "errors_no"),
-                 'filter' : {'class' : 'errors_no', 
-                             'label': 'Show:', 
+                 'filter' : {'class' : 'errors_no',
+                             'label': 'Show:',
                              'options' : [
                                          ('Builds with errors', 'errors_no__gte:1', queryset_with_search.filter(errors_no__gte=1).count()),
                                          ('Builds without errors', 'errors_no:0', queryset_with_search.filter(errors_no=0).count()),
@@ -319,8 +320,8 @@ def builds(request):
                  'qhelp': "How many warnigns were encountered during the build (if any)",
                  'orderfield': _get_toggle_order(request, "warnings_no", True),
                  'ordericon':_get_toggle_order_icon(request, "warnings_no"),
-                 'filter' : {'class' : 'warnings_no', 
-                             'label': 'Show:', 
+                 'filter' : {'class' : 'warnings_no',
+                             'label': 'Show:',
                              'options' : [
                                          ('Builds with warnings','warnings_no__gte:1', queryset_with_search.filter(warnings_no__gte=1).count()),
                                          ('Builds without warnings','warnings_no:0', queryset_with_search.filter(warnings_no=0).count()),
@@ -417,13 +418,236 @@ def recipe(request, build_id, recipe_id):
 
 def target(request, build_id, target_id):
     template = "target.html"
-    if Build.objects.filter(pk=build_id).count() == 0 :
-        return redirect(builds)
-    context = {
-            'build' : Build.objects.filter(pk=build_id)[0],
-    }
+    mandatory_parameters = { 'count': 25,  'page' : 1, 'orderby':'name:+'};
+    retval = _verify_parameters( request.GET, mandatory_parameters )
+    if retval:
+        return _redirect_parameters( 'target', request.GET, mandatory_parameters, build_id = build_id, target_id = target_id)
+    (filter_string, search_term, ordering_string) = _search_tuple(request, Package)
+
+    # FUTURE:  get rid of nested sub-queries replacing with ManyToMany field
+    queryset = Package.objects.filter(id__in=Target_Installed_Package.objects.filter(target_id=target_id).values('package_id'))
+    packages_sum =  queryset.aggregate(Sum('installed_size'))
+    queryset = _get_queryset(Package, queryset, filter_string, search_term, ordering_string)
+    packages = _build_page_range(Paginator(queryset, request.GET.get('count', 25)),request.GET.get('page', 1))
+    context = { 'build': Build.objects.filter(pk=build_id)[0],
+                'target': Target.objects.filter(pk=target_id)[0],
+                'objects': packages,
+                'packages_sum' : packages_sum['installed_size__sum'],
+                'object_search_display': "packages included",
+                'tablecols':[
+                {
+                    'name':'Package',
+                    'qhelp':'Packaged output resulting from building a recipe and included in this image',
+                    'orderfield': _get_toggle_order(request, "name"),
+                    'ordericon':_get_toggle_order_icon(request, "name"),
+                },
+                {
+                    'name':'Package version',
+                    'qhelp':'The package version and revision',
+                },
+                {
+                    'name':'Size',
+                    'qhelp':'The size of the package',
+                    'orderfield': _get_toggle_order(request, "size"),
+                    'ordericon':_get_toggle_order_icon(request, "size"),
+                    'clclass': 'package_size',
+                    'hidden' : 0,
+                },
+                {
+                    'name':'Size over total (%)',
+                    'qhelp':'Proportion of the overall included package size represented by this package',
+                    'orderfield': _get_toggle_order(request, "size"),
+                    'ordericon':_get_toggle_order_icon(request, "size"),
+                    'clclass': 'size_over_total',
+                    'hidden' : 1,
+                },
+                {
+                    'name':'License',
+                    'qhelp':'The license under which the package is distributed. Separate license names using | (pipe) means there is a choice between licenses. Separate license names using & (ampersand) means multiple licenses exist that cover different parts of the source',
+                    'orderfield': _get_toggle_order(request, "license"),
+                    'ordericon':_get_toggle_order_icon(request, "license"),
+                    'clclass': 'license',
+                    'hidden' : 1,
+                },
+                {
+                    'name':'Dependencies',
+                    'qhelp':"Package runtime dependencies (other packages)",
+                    'clclass': 'depends',
+                    'hidden' : 0,
+                },
+                {
+                    'name':'Reverse dependencies',
+                    'qhelp':'Package run-time reverse dependencies (i.e. which other packages depend on this package',
+                    'clclass': 'brought_in_by',
+                    'hidden' : 0,
+                },
+                {
+                    'name':'Recipe',
+                    'qhelp':'The name of the recipe building the package',
+                    'orderfield': _get_toggle_order(request, "recipe__name"),
+                    'ordericon':_get_toggle_order_icon(request, "recipe__name"),
+                    'clclass': 'recipe_name',
+                    'hidden' : 0,
+                },
+                {
+                    'name':'Recipe version',
+                    'qhelp':'Version and revision of the recipe building the package',
+                    'clclass': 'recipe_version',
+                    'hidden' : 1,
+                },
+                {
+                    'name':'Layer',
+                    'qhelp':'The name of the layer providing the recipe that builds the package',
+                    'orderfield': _get_toggle_order(request, "recipe__layer_version__layer__name"),
+                    'ordericon':_get_toggle_order_icon(request, "recipe__layer_version__layer__name"),
+                    'clclass': 'layer_name',
+                    'hidden' : 1,
+                },
+                {
+                    'name':'Layer branch',
+                    'qhelp':'The Git branch of the layer providing the recipe that builds the package',
+                    'orderfield': _get_toggle_order(request, "recipe__layer_version__branch"),
+                    'ordericon':_get_toggle_order_icon(request, "recipe__layer_version__branch"),
+                    'clclass': 'layer_branch',
+                    'hidden' : 1,
+                },
+                {
+                    'name':'Layer commit',
+                    'qhelp':'The Git commit of the layer providing the recipe that builds the package',
+                    'clclass': 'layer_commit',
+                    'hidden' : 1,
+                },
+                {
+                    'name':'Layer directory',
+                    'qhelp':'Location in disk of the layer providing the recipe that builds the package',
+                    'orderfield': _get_toggle_order(request, "recipe__layer_version__layer__local_path"),
+                    'ordericon':_get_toggle_order_icon(request, "recipe__layer_version__layer__local_path"),
+                    'clclass': 'layer_directory',
+                    'hidden' : 1,
+                },
+                ]
+        }
+
     return render(request, template, context)
 
+from django.core.serializers.json import DjangoJSONEncoder
+from django.http import HttpResponse
+def dirinfo_ajax(request, build_id, target_id):
+    top = request.GET.get('start', '/')
+    return HttpResponse(_get_dir_entries(build_id, target_id, top))
+
+from django.utils.functional import Promise
+from django.utils.encoding import force_text
+class LazyEncoder(json.JSONEncoder):
+    def default(self, obj):
+        if isinstance(obj, Promise):
+            return force_text(obj)
+        return super(LazyEncoder, self).default(obj)
+
+from toastergui.templatetags.projecttags import filtered_filesizeformat
+from django import template
+import os
+def _get_dir_entries(build_id, target_id, start):
+    node_str = {
+        Target_File.ITYPE_REGULAR   : '-',
+        Target_File.ITYPE_DIRECTORY : 'd',
+        Target_File.ITYPE_SYMLINK   : 'l',
+        Target_File.ITYPE_SOCKET    : 's',
+        Target_File.ITYPE_FIFO      : 'p',
+        Target_File.ITYPE_CHARACTER : 'c',
+        Target_File.ITYPE_BLOCK     : 'b',
+    }
+    response = []
+    objects  = Target_File.objects.filter(target__exact=target_id, directory__path=start)
+    target_packages = Target_Installed_Package.objects.filter(target__exact=target_id).values_list('package_id', flat=True)
+    for o in objects:
+        # exclude root inode '/'
+        if o.path == '/':
+            continue
+        try:
+            entry = {}
+            entry['parent'] = start
+            entry['name'] = os.path.basename(o.path)
+            entry['fullpath'] = o.path
+
+            # set defaults, not all dentries have packages
+            entry['installed_package'] = None
+            entry['package_id'] = None
+            entry['package'] = None
+            entry['link_to'] = None
+            if o.inodetype == Target_File.ITYPE_DIRECTORY:
+                entry['isdir'] = 1
+                # is there content in directory
+                entry['childcount'] = Target_File.objects.filter(directory__path=o.path).all().count()
+            else:
+                entry['isdir'] = 0
+
+                # resolve the file to get the package from the resolved file
+                resolved_id = o.sym_target_id
+                resolved_path = o.path
+                if target_packages.count():
+                    while resolved_id != "" and resolved_id != None:
+                        tf = Target_File.objects.get(pk=resolved_id)
+                        resolved_path = tf.path
+                        resolved_id = tf.sym_target_id
+
+                    thisfile=Package_File.objects.all().filter(path__exact=resolved_path, package_id__in=target_packages)
+                    if thisfile.count():
+                        p = Package.objects.get(pk=thisfile[0].package_id)
+                        entry['installed_package'] = p.installed_name
+                        entry['package_id'] = str(p.id)
+                        entry['package'] = p.name
+                # don't use resolved path from above, show immediate link-to
+                if o.sym_target_id != "" and o.sym_target_id != None:
+                    entry['link_to'] = Target_File.objects.get(pk=o.sym_target_id).path
+            t = template.Template('{% load projecttags %} {{ size|filtered_filesizeformat }}')
+            c = template.Context({'size': o.size})
+            entry['size'] = str(t.render(c))
+            if entry['link_to'] != None:
+                entry['permission'] = node_str[o.inodetype] + o.permission
+            else:
+                entry['permission'] = node_str[o.inodetype] + o.permission
+            entry['owner'] = o.owner
+            entry['group'] = o.group
+            response.append(entry)
+
+        except:
+            pass
+
+    # sort by directories first, then by name
+    rsorted = sorted(response, key=lambda entry :  entry['name'])
+    rsorted = sorted(rsorted, key=lambda entry :  entry['isdir'], reverse=True)
+    return json.dumps(rsorted, cls=LazyEncoder)
+
+def dirinfo(request, build_id, target_id, file_path=None):
+    template = "dirinfo.html"
+    objects = _get_dir_entries(build_id, target_id, '/')
+    packages_sum = Package.objects.filter(id__in=Target_Installed_Package.objects.filter(target_id=target_id).values('package_id')).aggregate(Sum('installed_size'))
+    dir_list = None
+    if file_path != None:
+        """
+        Link from the included package detail file list page and is
+        requesting opening the dir info to a specific file path.
+        Provide the list of directories to expand and the full path to
+        highlight in the page.
+        """
+        # Aassume target's path separator matches host's, that is, os.sep
+        sep = os.sep
+        dir_list = []
+        head = file_path
+        while head != sep:
+            (head,tail) = os.path.split(head)
+            if head != sep:
+                dir_list.insert(0, head)
+
+    context = { 'build': Build.objects.filter(pk=build_id)[0],
+                'target': Target.objects.filter(pk=target_id)[0],
+                'packages_sum': packages_sum['installed_size__sum'],
+                'objects': objects,
+                'dir_list': dir_list,
+                'file_path': file_path,
+              }
+    return render(request, template, context)
 
 def _find_task_dep(task):
     tp = []
@@ -593,7 +817,7 @@ def tasks_common(request, build_id, variant):
                    }
 
     }
-    #if   'tasks' == variant: tc_cache['hidden']='0'; 
+    #if   'tasks' == variant: tc_cache['hidden']='0';
     tc_time={
         'name':'Time (secs)',
         'qhelp':'How long it took the task to finish, expressed in seconds',
@@ -796,7 +1020,7 @@ def configvars(request, build_id):
     # remove duplicate records from multiple search hits in the VariableHistory table
     queryset = queryset.distinct()
     # remove records where the value is empty AND there are no history files
-    queryset = queryset.exclude(variable_value='',vhistory__file_name__isnull=True)    
+    queryset = queryset.exclude(variable_value='',vhistory__file_name__isnull=True)
 
     variables = _build_page_range(Paginator(queryset, request.GET.get('count', 50)), request.GET.get('page', 1))
 
@@ -811,7 +1035,7 @@ def configvars(request, build_id):
         file_filter += 'conf/distro/'
     if filter_string.find('/bitbake.conf') > 0:
         file_filter += '/bitbake.conf'
-    
+
     context = {
                 'objectname': 'configvars',
                 'object_search_display':'variables',