summaryrefslogtreecommitdiffstats
path: root/bitbake/lib/toaster
diff options
context:
space:
mode:
authorDave Lerner <dave.lerner@windriver.com>2014-03-12 16:54:09 -0500
committerRichard Purdie <richard.purdie@linuxfoundation.org>2014-03-21 14:47:53 +0000
commit4cdd56fff3471669ff2372f0d8bf15f9a287374a (patch)
tree76b81b2cb87e3b3df9751d8487351b5dfb2c59a3 /bitbake/lib/toaster
parente94f0b4e8c2cbea92688975b1a45a061af48911e (diff)
downloadpoky-4cdd56fff3471669ff2372f0d8bf15f9a287374a.tar.gz
bitbake: toaster: image information views
[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>
Diffstat (limited to 'bitbake/lib/toaster')
-rw-r--r--bitbake/lib/toaster/orm/models.py5
-rw-r--r--bitbake/lib/toaster/toastergui/static/css/default.css2
-rw-r--r--bitbake/lib/toaster/toastergui/static/css/jquery.treetable.theme.toaster.css38
-rw-r--r--bitbake/lib/toaster/toastergui/static/jquery.treetable.theme.toaster.css66
-rw-r--r--bitbake/lib/toaster/toastergui/templates/base.html2
-rw-r--r--bitbake/lib/toaster/toastergui/templates/basebuildpage.html2
-rw-r--r--bitbake/lib/toaster/toastergui/templates/dirinfo.html237
-rw-r--r--bitbake/lib/toaster/toastergui/templates/package_included_detail.html2
-rw-r--r--bitbake/lib/toaster/toastergui/templates/target.html153
-rw-r--r--bitbake/lib/toaster/toastergui/templatetags/projecttags.py49
-rw-r--r--bitbake/lib/toaster/toastergui/urls.py4
-rw-r--r--bitbake/lib/toaster/toastergui/views.py258
12 files changed, 792 insertions, 26 deletions
diff --git a/bitbake/lib/toaster/orm/models.py b/bitbake/lib/toaster/orm/models.py
index 93506d7c1b..c5fe69b883 100644
--- a/bitbake/lib/toaster/orm/models.py
+++ b/bitbake/lib/toaster/orm/models.py
@@ -58,6 +58,9 @@ class Target(models.Model):
58 image_size = models.IntegerField(default=0) 58 image_size = models.IntegerField(default=0)
59 license_manifest_path = models.CharField(max_length=500, null=True) 59 license_manifest_path = models.CharField(max_length=500, null=True)
60 60
61 def package_count(self):
62 return Target_Installed_Package.objects.filter(target_id__exact=self.id).count()
63
61 def __str__(self): 64 def __str__(self):
62 return self.target 65 return self.target
63 66
@@ -194,7 +197,7 @@ class Task_Dependency(models.Model):
194 depends_on = models.ForeignKey(Task, related_name='task_dependencies_depends') 197 depends_on = models.ForeignKey(Task, related_name='task_dependencies_depends')
195 198
196class Package(models.Model): 199class Package(models.Model):
197 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'] 200 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']
198 build = models.ForeignKey('Build') 201 build = models.ForeignKey('Build')
199 recipe = models.ForeignKey('Recipe', null=True) 202 recipe = models.ForeignKey('Recipe', null=True)
200 name = models.CharField(max_length=100) 203 name = models.CharField(max_length=100)
diff --git a/bitbake/lib/toaster/toastergui/static/css/default.css b/bitbake/lib/toaster/toastergui/static/css/default.css
index 7db156a16e..53a3fee19f 100644
--- a/bitbake/lib/toaster/toastergui/static/css/default.css
+++ b/bitbake/lib/toaster/toastergui/static/css/default.css
@@ -104,7 +104,7 @@ select { width: auto; }
104.well > .lead, .alert .lead { margin-bottom: 0px; } 104.well > .lead, .alert .lead { margin-bottom: 0px; }
105.no-results { margin: 10px 0; } 105.no-results { margin: 10px 0; }
106.task-name { margin-left: 7px; } 106.task-name { margin-left: 7px; }
107 107.icon-hand-right {color: #ccccc; }
108 108
109 109
110 110
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
index 0000000000..d8552e5816
--- /dev/null
+++ b/bitbake/lib/toaster/toastergui/static/css/jquery.treetable.theme.toaster.css
@@ -0,0 +1,38 @@
1table.treetable span.file {
2 background-image: url();
3}
4
5table.treetable span.folder {
6 background-image: url();
7}
8
9table.treetable tr.collapsed span.indenter a {
10 background-image: url();
11}
12
13table.treetable tr.expanded span.indenter a {
14 background-image: url();
15}
16
17
18
19table.treetable tr.collapsed.selected span.indenter a {
20 background-image: url();
21}
22
23table.treetable tr.expanded.selected span.indenter a {
24 background-image: url();
25}
26
27table.treetable tr.accept {
28 background-color: #a3bce4;
29 color: #fff
30}
31
32table.treetable tr.collapsed.accept td span.indenter a {
33 background-image: url();
34}
35
36table.treetable tr.expanded.accept td span.indenter a {
37 background-image: url();
38}
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
index 0000000000..5194b234d7
--- /dev/null
+++ b/bitbake/lib/toaster/toastergui/static/jquery.treetable.theme.toaster.css
@@ -0,0 +1,66 @@
1/*
2table.treetable {
3 border: 1px solid #888;
4 border-collapse: collapse;
5 font-size: .8em;
6 line-height: 1;
7 margin: .6em 0 1.8em 0;
8 width: 100%;
9}
10
11table.treetable caption {
12 font-size: .9em;
13 font-weight: bold;
14 margin-bottom: .2em;
15}
16
17table.treetable tbody tr td {
18 cursor: default;
19 padding: .3em 1em;
20}
21
22table.treetable span {
23 background-position: center left;
24 background-repeat: no-repeat;
25 padding: .2em 0 .2em 1.5em;
26}
27*/
28
29table.treetable span.file {
30 background-image: url();
31}
32
33table.treetable span.folder {
34 background-image: url();
35}
36
37table.treetable tr.collapsed span.indenter a {
38 background-image: url();
39}
40
41table.treetable tr.expanded span.indenter a {
42 background-image: url();
43}
44
45
46
47table.treetable tr.collapsed.selected span.indenter a {
48 background-image: url();
49}
50
51table.treetable tr.expanded.selected span.indenter a {
52 background-image: url();
53}
54
55table.treetable tr.accept {
56 background-color: #a3bce4;
57 color: #fff
58}
59
60table.treetable tr.collapsed.accept td span.indenter a {
61 background-image: url();
62}
63
64table.treetable tr.expanded.accept td span.indenter a {
65 background-image: url();
66}
diff --git a/bitbake/lib/toaster/toastergui/templates/base.html b/bitbake/lib/toaster/toastergui/templates/base.html
index 5493e230b1..9ca9c9ac3b 100644
--- a/bitbake/lib/toaster/toastergui/templates/base.html
+++ b/bitbake/lib/toaster/toastergui/templates/base.html
@@ -49,6 +49,8 @@ function reload_params(params) {
49} 49}
50</script> 50</script>
51 51
52{% block extraheadcontent %}
53{% endblock %}
52 </head> 54 </head>
53 55
54<body style="height: 100%"> 56<body style="height: 100%">
diff --git a/bitbake/lib/toaster/toastergui/templates/basebuildpage.html b/bitbake/lib/toaster/toastergui/templates/basebuildpage.html
index 054a37cc8c..636fca28dd 100644
--- a/bitbake/lib/toaster/toastergui/templates/basebuildpage.html
+++ b/bitbake/lib/toaster/toastergui/templates/basebuildpage.html
@@ -26,7 +26,7 @@
26 <div id="nav" class="span2"> 26 <div id="nav" class="span2">
27 <ul class="nav nav-list well"> 27 <ul class="nav nav-list well">
28 <li class="nav-header">Images</li> 28 <li class="nav-header">Images</li>
29 {% for t in build.target_set.all %} 29 {% for t in build.target_set.all|dictsort:"target" %}
30 <li><a href="{% url 'target' build.pk t.pk %}">{{t.target}}</a><li> 30 <li><a href="{% url 'target' build.pk t.pk %}">{{t.target}}</a><li>
31 {% endfor %} 31 {% endfor %}
32 <li class="nav-header">Build</li> 32 <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
index 0000000000..9b76a1cb8c
--- /dev/null
+++ b/bitbake/lib/toaster/toastergui/templates/dirinfo.html
@@ -0,0 +1,237 @@
1{% extends "basebuildpage.html" %}
2{% block extraheadcontent %}
3{% load static %}
4<link rel="stylesheet" href="{% static 'css/jquery.treetable.css' %}" type="text/css">
5<link rel="stylesheet" href="{% static 'css/jquery.treetable.theme.toaster.css' %}" type="text/css">
6{% endblock extraheadcontent %}
7
8{% block localbreadcrumb %}
9<li>{{target.target}}</li>
10{% endblock localbreadcrumb%}
11
12{% block buildinfomain %}
13
14{% load static %}
15<script src="{% static 'js/jquery.treetable.js' %}">
16</script>
17{% load projecttags %}
18
19<script type='text/javascript'>
20 function setupTreetable() {
21 $("#dirtable").treetable({
22 expandable: true,
23 branchAttr: "ttBranch",
24 clickableNodeNames: true,
25 onNodeCollapse: function() {
26 /* Do nothing, keep cached */
27 },
28 onNodeExpand: function() {
29 var start = this.id;
30 var n = $("#dirtable").treetable("node", start);
31 if (this.children.length > 0) {
32 /* already was expanded once */
33 $("#dirtable").treetable("reveal", start);
34 }
35 else {
36 var url = "{% url "dirinfo_ajax" build.id target.id %}";
37 $.ajax({
38 async: false,
39 type : "GET",
40 url : url,
41 data : "start=" + start,
42 success : function(response) {
43 var objects = $.parseJSON(response);
44 addRows(n, objects)
45 },
46 error : function(jqXHR, textStatus, errorThrown ) {alert(textStatus + ":" + errorThrown)},
47 });
48 }
49 },
50 });
51 }
52 function td(data) {
53 if (data == null) {
54 data = '';
55 }
56 return '<td>' + data + '</td>'
57 }
58
59 function formatRow(o) {
60 /* setup tr-wide formatting */
61 var tr = '<tr class="';
62 if (o.link_to != null) {
63 tr += 'muted ';
64 }
65 if (o.isdir && o.childcount) {
66 tr += 'branch" data-tt-branch="true" ';
67 }
68 else {
69 tr += 'leaf" data-tt-branch="false" ';
70 }
71 tr += ' data-tt-id="' + o.fullpath +'" ';
72 if (o.parent != "/") {
73 tr += ' data-tt-parent-id="' + o.parent +'" ';
74 }
75 tr += '>';
76
77 /* setup td specific formatting */
78 var link_to = td(o.link_to);
79 var size = td(o.size);
80 var permission = td(o.permission);
81 var owner = td(o.owner);
82 var group = td(o.group);
83
84 /* handle the name column */
85 var name = null;;
86 var namespan=1;
87 if (o.isdir) {
88 if (o.link_to == null) {
89 namespan = 2;
90 if (o.package == null) {
91 namespan = 3;
92 }
93 }
94 var colspan = 'colspan="' + namespan + '"';
95 name = '<td class="content-directory"' + colspan + '>';
96 if (o.childcount) {
97 name += '<a href="">';
98 }
99 name += '<i class="icon-folder-close"></i>';
100 name += '&nbsp;' + o.name;
101 if (o.childcount) {
102 name += '</a>';
103 }
104 name += '</td>';
105 }
106 else {
107 name = '<td>';
108 if (o.link_to == null) {
109 name += '<i class="icon-file"></i>';
110 }
111 else {
112 name += '<i class="icon-hand-right"></i>';
113 }
114 name += '&nbsp;' + o.name;
115 name += '</td>';
116 }
117
118 /* handle the package column */
119 var package = null;
120 if (o.package != null) {
121 /* add link to included package page */
122 build_id = {{ build.id }};
123 target_id = {{ target.id }};
124 /* Create a url for a dummy package id of 0 */
125 dummy = "{% url 'package_included_detail' build.id target.id 0 %}"
126 /* fill in the package id */
127 url = dummy.substr(0, dummy.length-1) + o.package_id;
128 package = '<a href=' + url + '>' ;
129 package += o.package;
130 package += '</a>';
131 if (o.installed_package != o.package) {
132 /* make class muted and add hover help */
133 package += '<span class="muted"> as ' + o.installed_package + ' </span>';
134 package += '<i class="icon-question-sign get-help hover-help" ';
135 package += 'title="' + o.package + ' was renamed at packaging time and was installed in your image as ' + o.installed_package + '">';
136 package += '</i>';
137 }
138 }
139 package = td(package);
140
141 var cols1to3;
142 switch (namespan) {
143 case 3:
144 cols1to3 = name;
145 break;
146 case 2:
147 cols1to3 = name + package;
148 break;
149 default:
150 cols1to3 = name + link_to + package;
151 }
152 r = tr + cols1to3 + size + permission + owner + group + "</tr>"
153 return r;
154 }
155
156 function addRows(n, objs) {
157 rows = "";
158 for (i=0; i<objs.length; i++) {
159 rows += formatRow(objs[i]);
160 }
161 $("#dirtable").treetable("loadBranch", n, rows);
162 }
163
164 $.fn.isOffScreen = function(){
165 var win = $(window);
166 viewportBottom = win.scrollTop() + win.height();
167
168 var bounds = this.offset();
169 bounds.bottom = bounds.top + this.outerHeight();
170
171 return (bounds.bottom > viewportBottom);
172 };
173
174 function selectRow(path) {
175 var row = $('tr[data-tt-id="' + path + '"]');
176 row.addClass(" highlight");
177 if (row.isOffScreen()) {
178 $('html, body').animate({ scrollTop: row.offset().top - 150}, 2000);
179 }
180 }
181</script>
182
183<div class="span10">
184
185 <div class="page-header">
186 <h1> {{target.target}} </h1>
187 </div>
188
189 <ul class="nav nav-pills">
190 <li class="">
191 <a href="{% url 'target' build.id target.id %}">
192 <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>
193 Packages included ({{target.package_count}} - {{packages_sum|filtered_filesizeformat}})
194 </a>
195 </li>
196 <li class="active">
197 <a href="{% url 'dirinfo' build.id target.id %}">
198 <i class="icon-question-sign get-help" data-toggle="tooltip" title="The directories and files in the root file system of this image"></i>
199 Directory structure
200 </a>
201 </li>
202 </ul>
203
204 <div id="directory-structure" class="tab-pane active">
205 <table id="dirtable" class="table table-bordered table-hover treetable">
206 <thead>
207 <tr>
208 <th>Directory / File</th>
209 <th>Symbolic link to</th>
210 <th>Source package</th>
211 <th>Size</th>
212 <th>Permissions</th>
213 <th>Owner</th>
214 <th>Group</th>
215 </tr>
216 </thead>
217 <tbody>
218 <script type='text/javascript'>
219 setupTreetable();
220 addRows(null, {{ objects|safe }} );
221 {% if file_path %}
222 {% comment %}
223 link from package_included_detail specifies file path
224 {% endcomment %}
225 {% for dir_elem in dir_list %}
226 $("#dirtable").treetable("expandNode", "{{dir_elem}}");
227 {% endfor %}
228 selectRow("{{file_path}}");
229 {% endif %}
230 </script>
231 </tbody>
232 </table>
233 </div> <!-- directory-structure -->
234</div> <!-- span10 -->
235
236{% endblock buildinfomain %}
237
diff --git a/bitbake/lib/toaster/toastergui/templates/package_included_detail.html b/bitbake/lib/toaster/toastergui/templates/package_included_detail.html
index df2588548c..ce4f1cb33c 100644
--- a/bitbake/lib/toaster/toastergui/templates/package_included_detail.html
+++ b/bitbake/lib/toaster/toastergui/templates/package_included_detail.html
@@ -24,7 +24,7 @@
24 {% for file in package.buildfilelist_package.all|dictsort:"path" %} 24 {% for file in package.buildfilelist_package.all|dictsort:"path" %}
25 <tr> 25 <tr>
26 <td> 26 <td>
27 <a href="{% url 'image_information_dir' build.id target.id file.id %}"> 27 <a href="{% url 'dirinfo_filepath' build.id target.id file.path %}">
28 {{file.path}} 28 {{file.path}}
29 </a> 29 </a>
30 </td> 30 </td>
diff --git a/bitbake/lib/toaster/toastergui/templates/target.html b/bitbake/lib/toaster/toastergui/templates/target.html
index f2d0ad461b..45128986e1 100644
--- a/bitbake/lib/toaster/toastergui/templates/target.html
+++ b/bitbake/lib/toaster/toastergui/templates/target.html
@@ -1,8 +1,153 @@
1{% extends "basebuildpage.html" %} 1{% extends "basebuildpage.html" %}
2
3{% block localbreadcrumb %} 2{% block localbreadcrumb %}
4<li>Target</li> 3<li>{{target.target}}</li>
5{% endblock %} 4{% endblock localbreadcrumb%}
5
6{% load projecttags %}
6 7
7{% block buildinfomain %} 8{% block buildinfomain %}
8{% endblock %} 9
10<div class="row-fluid span10">
11 <div class="page-header">
12 <h1>
13 {% if request.GET.search and objects.paginator.count > 0 %}
14 {{objects.paginator.count}} package{{objects.paginator.count|pluralize}} found
15 {% elif request.GET.search and objects.paginator.count == 0 %}
16 No packages found
17 {% else %}
18 {{target.target}}
19 {% endif %}
20 </h1>
21 </div>
22</div>
23
24<div class="row-fluid pull-right span10" id="navTab">
25 <ul class="nav nav-pills">
26 <li class="active">
27 <a href="#target">
28 <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>
29 Packages included ({{target.package_count}} - {{packages_sum|filtered_filesizeformat}})
30 </a>
31 </li>
32 <li>
33 <a href="{% url 'dirinfo' build.id target.id %}">
34 <i class="icon-question-sign get-help" data-toggle="tooltip" title="The directories and files in the root file system of this image"></i>
35 Directory structure
36 </a>
37 </li>
38 </ul>
39
40 <div id="image-packages" class="tab-pane">
41
42 {% if objects.paginator.count == 0 %}
43 <div class="row-fluid">
44 <div class="alert">
45 <form class="no-results input-append" id="searchform">
46 <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 %}
47 <button class="btn" type="submit" value="Search">Search</button>
48 <button class="btn btn-link" onclick="javascript:$('#search').val('');searchform.submit()">Show all packages</button>
49 </form>
50 </div>
51 </div>
52
53
54 {% else %}
55 {% include "basetable_top.html" %}
56 {% for package in objects %}
57 <tr>
58 <td class="package_name">
59 <a href="{% url 'package_included_detail' build.id target.id package.id %}">
60 {{package.name}}
61 </a>
62 {% if package.installed_name and package.name != package.installed_name %}
63 <span class="muted"> as {{package.installed_name}}</span>
64 <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>
65 {% endif %}
66 </td>
67 <td class="package_version">
68 <a href="{% url 'package_included_detail' build.id target.id package.id %}">
69 {{package.version|filtered_packageversion:package.revision}}
70 </a>
71 </td>
72 <td class="package_size">
73 {{package.size|filtered_installedsize:package.installed_size|filtered_filesizeformat}}
74 </td>
75 <td class="size_over_total">
76 {{package|filter_sizeovertotal:packages_sum}}
77 </td>
78 <td class="license">
79 {{package.license}}
80 </td>
81 <td class="depends">
82 {% with deps=package|runtime_dependencies:target.id %}
83 {% with deps_count=deps|length %}
84 {% if deps_count > 0 %}
85 <a class="btn"
86 title="<a href='{% url "package_included_dependencies" build.id target.id package.id %}'>{{package.name}}</a> depends on"
87 data-content="<ul class='unstyled'>
88 {% for i in deps|dictsort:'depends_on.name' %}
89 <li><a href='{% url "package_included_dependencies" build.pk target.id i.depends_on.pk %}'>{{i.depends_on.name}}</a></li>
90 {% endfor %}
91 </ul>">
92 {{deps_count}}
93 </a>
94 {% endif %}
95 {% endwith %}
96 {% endwith %}
97 </td>
98 <td class="brought_in_by">
99 {% with rdeps=package|reverse_runtime_dependencies:target.id %}
100 {% with rdeps_count=rdeps|length %}
101 {% if rdeps_count > 0 %}
102 <a class="btn"
103 title="<a href='{% url "package_included_reverse_dependencies" build.id target.id package.id %}'>{{package.name}}</a> is brought in by"
104 data-content="<ul class='unstyled'>
105 {% for i in rdeps|dictsort:'package.name' %}
106 <li><a href='{% url "package_included_dependencies" build.id target.id i.package.id %}'>{{i.package.name}}</a></li>
107 {% endfor %}
108 </ul>">
109 {{rdeps_count}}
110 </a>
111 {% endif %}
112 {% endwith %}
113 {% endwith %}
114 </td>
115 <td class="recipe_name">
116 {% if package.recipe.version %}
117 <a href="{% url 'recipe' build.id package.recipe_id %}">
118 {{ package.recipe.name }}
119 </a>
120 {% endif %}
121 </td>
122 <td class="recipe_version">
123 {% if package.recipe.version %}
124 <a href="{% url 'recipe' build.id package.recipe_id %}">
125 {{ package.recipe.version }}
126 </a>
127 {% endif %}
128 </td>
129 <td class="layer_name">
130 {{ package.recipe.layer_version.layer.name }}
131 </td>
132 <td class="layer_branch">
133 {{ package.recipe.layer_version.branch}}
134 </td>
135 <td class="layer_commit">
136 <a class="btn"
137 data-content="<ul class='unstyled'>
138 <li>{{package.recipe.layer_version.commit}}</li>
139 </ul>">
140 {{package.recipe.layer_version.commit|truncatechars:13}}
141 </a>
142 </td>
143 <td class="layer_directory">
144 {{ package.recipe.layer_version.layer.local_path }}
145 </td>
146 </tr>
147 {% endfor %}
148
149 {% include "basetable_bottom.html" %}
150 {% endif %}
151 </div> <!-- tabpane -->
152</div> <!--span 10-->
153{% endblock buildinfomain %}
diff --git a/bitbake/lib/toaster/toastergui/templatetags/projecttags.py b/bitbake/lib/toaster/toastergui/templatetags/projecttags.py
index 60d5dd0b7c..e08258b6e7 100644
--- a/bitbake/lib/toaster/toastergui/templatetags/projecttags.py
+++ b/bitbake/lib/toaster/toastergui/templatetags/projecttags.py
@@ -215,3 +215,52 @@ def get_image_extensions( build ):
215 comma = ", "; 215 comma = ", ";
216 return( extensions ); 216 return( extensions );
217 217
218@register.filter
219def filtered_installedsize(size, installed_size):
220 """If package.installed_size not null and not empty return it,
221 else return package.size
222 """
223 return size if (installed_size == 0) or (installed_size == "") or (installed_size == None) else installed_size
224
225@register.filter
226def filtered_installedname(name, installed_name):
227 """If package.installed_name not null and not empty
228 return <div class=muted> as {{package.installed_name}}
229 otherwise ""
230 """
231 return name if (name == installed_name) or (not installed_name) or (installed_name == "") else name + " as " + installed_name
232
233@register.filter
234def filtered_packageversion(version, revision):
235 """ Emit "version-revision" if version and revision are not null
236 else "version" if version is not null
237 else ""
238 """
239 return "" if (not version or version == "") else version if (not revision or revision == "") else version + "-" + revision
240
241from django.db import models
242from orm.models import Package
243@register.filter
244def runtime_dependencies(package_object, targetid):
245 """ Return a queryset that lists the packages this package depends on
246 """
247 return package_object.package_dependencies_source.filter(target_id__exact=targetid, dep_type__in={'1'})
248
249@register.filter
250def reverse_runtime_dependencies(package_object, targetid):
251 """ Return a queryset that lists the packages depending on this package
252 """
253 return package_object.package_dependencies_target.filter(target_id__exact = targetid,dep_type__in={'1'})
254
255@register.filter
256def filter_sizeovertotal(package_object, total_size):
257 """ Return the % size of the package over the total size argument
258 formatted nicely.
259 """
260 size = package_object.installed_size
261 if size == None or size == '':
262 size = package_object.size
263
264 return '{:.1%}'.format(float(size)/float(total_size))
265
266
diff --git a/bitbake/lib/toaster/toastergui/urls.py b/bitbake/lib/toaster/toastergui/urls.py
index 8be27b08bc..ac83b387c5 100644
--- a/bitbake/lib/toaster/toastergui/urls.py
+++ b/bitbake/lib/toaster/toastergui/urls.py
@@ -45,8 +45,10 @@ urlpatterns = patterns('toastergui.views',
45 45
46 # images are known as targets in the internal model 46 # images are known as targets in the internal model
47 url(r'^build/(?P<build_id>\d+)/target/(?P<target_id>\d+)$', 'target', name='target'), 47 url(r'^build/(?P<build_id>\d+)/target/(?P<target_id>\d+)$', 'target', name='target'),
48 url(r'^dentries/build/(?P<build_id>\d+)/target/(?P<target_id>\d+)$', 'dirinfo_ajax', name='dirinfo_ajax'),
49 url(r'^build/(?P<build_id>\d+)/target/(?P<target_id>\d+)/dirinfo$', 'dirinfo', name='dirinfo'),
50 url(r'^build/(?P<build_id>\d+)/target/(?P<target_id>\d+)/dirinfo_filepath/(?P<file_path>(?:/[^/\n]+)*)$', 'dirinfo', name='dirinfo_filepath'),
48 url(r'^build/(?P<build_id>\d+)/target/(?P<target_id>\d+)/packages$', 'tpackage', name='targetpackages'), 51 url(r'^build/(?P<build_id>\d+)/target/(?P<target_id>\d+)/packages$', 'tpackage', name='targetpackages'),
49
50 url(r'^build/(?P<build_id>\d+)/configuration$', 'configuration', name='configuration'), 52 url(r'^build/(?P<build_id>\d+)/configuration$', 'configuration', name='configuration'),
51 url(r'^build/(?P<build_id>\d+)/configvars$', 'configvars', name='configvars'), 53 url(r'^build/(?P<build_id>\d+)/configvars$', 'configvars', name='configvars'),
52 url(r'^build/(?P<build_id>\d+)/buildtime$', 'buildtime', name='buildtime'), 54 url(r'^build/(?P<build_id>\d+)/buildtime$', 'buildtime', name='buildtime'),
diff --git a/bitbake/lib/toaster/toastergui/views.py b/bitbake/lib/toaster/toastergui/views.py
index 9740ef38d1..97514cc0f6 100644
--- a/bitbake/lib/toaster/toastergui/views.py
+++ b/bitbake/lib/toaster/toastergui/views.py
@@ -21,17 +21,18 @@
21 21
22import operator 22import operator
23 23
24from django.db.models import Q 24from django.db.models import Q, Sum
25from django.shortcuts import render, redirect 25from django.shortcuts import render, redirect
26from orm.models import Build, Target, Task, Layer, Layer_Version, Recipe, LogMessage, Variable 26from orm.models import Build, Target, Task, Layer, Layer_Version, Recipe, LogMessage, Variable
27from orm.models import Task_Dependency, Recipe_Dependency, Package, Package_File, Package_Dependency 27from orm.models import Task_Dependency, Recipe_Dependency, Package, Package_File, Package_Dependency
28from orm.models import Target_Installed_Package 28from orm.models import Target_Installed_Package, Target_File
29from django.views.decorators.cache import cache_control 29from django.views.decorators.cache import cache_control
30from django.core.paginator import Paginator, EmptyPage, PageNotAnInteger 30from django.core.paginator import Paginator, EmptyPage, PageNotAnInteger
31from django.http import HttpResponseBadRequest 31from django.http import HttpResponseBadRequest
32from django.utils import timezone 32from django.utils import timezone
33from datetime import timedelta 33from datetime import timedelta
34from django.utils import formats 34from django.utils import formats
35import json
35 36
36def _build_page_range(paginator, index = 1): 37def _build_page_range(paginator, index = 1):
37 try: 38 try:
@@ -163,7 +164,7 @@ def _get_search_results(search_term, queryset, model):
163def _search_tuple(request, model): 164def _search_tuple(request, model):
164 ordering_string, invalid = _validate_input(request.GET.get('orderby', ''), model) 165 ordering_string, invalid = _validate_input(request.GET.get('orderby', ''), model)
165 if invalid: 166 if invalid:
166 raise BaseException("Invalid ordering " + str(invalid)) 167 raise BaseException("Invalid ordering model:" + str(model) + str(invalid))
167 168
168 filter_string, invalid = _validate_input(request.GET.get('filter', ''), model) 169 filter_string, invalid = _validate_input(request.GET.get('filter', ''), model)
169 if invalid: 170 if invalid:
@@ -284,8 +285,8 @@ def builds(request):
284 'qhelp': "The date and time the build finished", 285 'qhelp': "The date and time the build finished",
285 'orderfield': _get_toggle_order(request, "completed_on", True), 286 'orderfield': _get_toggle_order(request, "completed_on", True),
286 'ordericon':_get_toggle_order_icon(request, "completed_on"), 287 'ordericon':_get_toggle_order_icon(request, "completed_on"),
287 'filter' : {'class' : 'completed_on', 288 'filter' : {'class' : 'completed_on',
288 'label': 'Show:', 289 'label': 'Show:',
289 'options' : [ 290 'options' : [
290 ("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()), 291 ("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()),
291 ("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()), 292 ("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):
307 'qhelp': "How many errors were encountered during the build (if any)", 308 'qhelp': "How many errors were encountered during the build (if any)",
308 'orderfield': _get_toggle_order(request, "errors_no", True), 309 'orderfield': _get_toggle_order(request, "errors_no", True),
309 'ordericon':_get_toggle_order_icon(request, "errors_no"), 310 'ordericon':_get_toggle_order_icon(request, "errors_no"),
310 'filter' : {'class' : 'errors_no', 311 'filter' : {'class' : 'errors_no',
311 'label': 'Show:', 312 'label': 'Show:',
312 'options' : [ 313 'options' : [
313 ('Builds with errors', 'errors_no__gte:1', queryset_with_search.filter(errors_no__gte=1).count()), 314 ('Builds with errors', 'errors_no__gte:1', queryset_with_search.filter(errors_no__gte=1).count()),
314 ('Builds without errors', 'errors_no:0', queryset_with_search.filter(errors_no=0).count()), 315 ('Builds without errors', 'errors_no:0', queryset_with_search.filter(errors_no=0).count()),
@@ -319,8 +320,8 @@ def builds(request):
319 'qhelp': "How many warnigns were encountered during the build (if any)", 320 'qhelp': "How many warnigns were encountered during the build (if any)",
320 'orderfield': _get_toggle_order(request, "warnings_no", True), 321 'orderfield': _get_toggle_order(request, "warnings_no", True),
321 'ordericon':_get_toggle_order_icon(request, "warnings_no"), 322 'ordericon':_get_toggle_order_icon(request, "warnings_no"),
322 'filter' : {'class' : 'warnings_no', 323 'filter' : {'class' : 'warnings_no',
323 'label': 'Show:', 324 'label': 'Show:',
324 'options' : [ 325 'options' : [
325 ('Builds with warnings','warnings_no__gte:1', queryset_with_search.filter(warnings_no__gte=1).count()), 326 ('Builds with warnings','warnings_no__gte:1', queryset_with_search.filter(warnings_no__gte=1).count()),
326 ('Builds without warnings','warnings_no:0', queryset_with_search.filter(warnings_no=0).count()), 327 ('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):
417 418
418def target(request, build_id, target_id): 419def target(request, build_id, target_id):
419 template = "target.html" 420 template = "target.html"
420 if Build.objects.filter(pk=build_id).count() == 0 : 421 mandatory_parameters = { 'count': 25, 'page' : 1, 'orderby':'name:+'};
421 return redirect(builds) 422 retval = _verify_parameters( request.GET, mandatory_parameters )
422 context = { 423 if retval:
423 'build' : Build.objects.filter(pk=build_id)[0], 424 return _redirect_parameters( 'target', request.GET, mandatory_parameters, build_id = build_id, target_id = target_id)
424 } 425 (filter_string, search_term, ordering_string) = _search_tuple(request, Package)
426
427 # FUTURE: get rid of nested sub-queries replacing with ManyToMany field
428 queryset = Package.objects.filter(id__in=Target_Installed_Package.objects.filter(target_id=target_id).values('package_id'))
429 packages_sum = queryset.aggregate(Sum('installed_size'))
430 queryset = _get_queryset(Package, queryset, filter_string, search_term, ordering_string)
431 packages = _build_page_range(Paginator(queryset, request.GET.get('count', 25)),request.GET.get('page', 1))
432 context = { 'build': Build.objects.filter(pk=build_id)[0],
433 'target': Target.objects.filter(pk=target_id)[0],
434 'objects': packages,
435 'packages_sum' : packages_sum['installed_size__sum'],
436 'object_search_display': "packages included",
437 'tablecols':[
438 {
439 'name':'Package',
440 'qhelp':'Packaged output resulting from building a recipe and included in this image',
441 'orderfield': _get_toggle_order(request, "name"),
442 'ordericon':_get_toggle_order_icon(request, "name"),
443 },
444 {
445 'name':'Package version',
446 'qhelp':'The package version and revision',
447 },
448 {
449 'name':'Size',
450 'qhelp':'The size of the package',
451 'orderfield': _get_toggle_order(request, "size"),
452 'ordericon':_get_toggle_order_icon(request, "size"),
453 'clclass': 'package_size',
454 'hidden' : 0,
455 },
456 {
457 'name':'Size over total (%)',
458 'qhelp':'Proportion of the overall included package size represented by this package',
459 'orderfield': _get_toggle_order(request, "size"),
460 'ordericon':_get_toggle_order_icon(request, "size"),
461 'clclass': 'size_over_total',
462 'hidden' : 1,
463 },
464 {
465 'name':'License',
466 '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',
467 'orderfield': _get_toggle_order(request, "license"),
468 'ordericon':_get_toggle_order_icon(request, "license"),
469 'clclass': 'license',
470 'hidden' : 1,
471 },
472 {
473 'name':'Dependencies',
474 'qhelp':"Package runtime dependencies (other packages)",
475 'clclass': 'depends',
476 'hidden' : 0,
477 },
478 {
479 'name':'Reverse dependencies',
480 'qhelp':'Package run-time reverse dependencies (i.e. which other packages depend on this package',
481 'clclass': 'brought_in_by',
482 'hidden' : 0,
483 },
484 {
485 'name':'Recipe',
486 'qhelp':'The name of the recipe building the package',
487 'orderfield': _get_toggle_order(request, "recipe__name"),
488 'ordericon':_get_toggle_order_icon(request, "recipe__name"),
489 'clclass': 'recipe_name',
490 'hidden' : 0,
491 },
492 {
493 'name':'Recipe version',
494 'qhelp':'Version and revision of the recipe building the package',
495 'clclass': 'recipe_version',
496 'hidden' : 1,
497 },
498 {
499 'name':'Layer',
500 'qhelp':'The name of the layer providing the recipe that builds the package',
501 'orderfield': _get_toggle_order(request, "recipe__layer_version__layer__name"),
502 'ordericon':_get_toggle_order_icon(request, "recipe__layer_version__layer__name"),
503 'clclass': 'layer_name',
504 'hidden' : 1,
505 },
506 {
507 'name':'Layer branch',
508 'qhelp':'The Git branch of the layer providing the recipe that builds the package',
509 'orderfield': _get_toggle_order(request, "recipe__layer_version__branch"),
510 'ordericon':_get_toggle_order_icon(request, "recipe__layer_version__branch"),
511 'clclass': 'layer_branch',
512 'hidden' : 1,
513 },
514 {
515 'name':'Layer commit',
516 'qhelp':'The Git commit of the layer providing the recipe that builds the package',
517 'clclass': 'layer_commit',
518 'hidden' : 1,
519 },
520 {
521 'name':'Layer directory',
522 'qhelp':'Location in disk of the layer providing the recipe that builds the package',
523 'orderfield': _get_toggle_order(request, "recipe__layer_version__layer__local_path"),
524 'ordericon':_get_toggle_order_icon(request, "recipe__layer_version__layer__local_path"),
525 'clclass': 'layer_directory',
526 'hidden' : 1,
527 },
528 ]
529 }
530
425 return render(request, template, context) 531 return render(request, template, context)
426 532
533from django.core.serializers.json import DjangoJSONEncoder
534from django.http import HttpResponse
535def dirinfo_ajax(request, build_id, target_id):
536 top = request.GET.get('start', '/')
537 return HttpResponse(_get_dir_entries(build_id, target_id, top))
538
539from django.utils.functional import Promise
540from django.utils.encoding import force_text
541class LazyEncoder(json.JSONEncoder):
542 def default(self, obj):
543 if isinstance(obj, Promise):
544 return force_text(obj)
545 return super(LazyEncoder, self).default(obj)
546
547from toastergui.templatetags.projecttags import filtered_filesizeformat
548from django import template
549import os
550def _get_dir_entries(build_id, target_id, start):
551 node_str = {
552 Target_File.ITYPE_REGULAR : '-',
553 Target_File.ITYPE_DIRECTORY : 'd',
554 Target_File.ITYPE_SYMLINK : 'l',
555 Target_File.ITYPE_SOCKET : 's',
556 Target_File.ITYPE_FIFO : 'p',
557 Target_File.ITYPE_CHARACTER : 'c',
558 Target_File.ITYPE_BLOCK : 'b',
559 }
560 response = []
561 objects = Target_File.objects.filter(target__exact=target_id, directory__path=start)
562 target_packages = Target_Installed_Package.objects.filter(target__exact=target_id).values_list('package_id', flat=True)
563 for o in objects:
564 # exclude root inode '/'
565 if o.path == '/':
566 continue
567 try:
568 entry = {}
569 entry['parent'] = start
570 entry['name'] = os.path.basename(o.path)
571 entry['fullpath'] = o.path
572
573 # set defaults, not all dentries have packages
574 entry['installed_package'] = None
575 entry['package_id'] = None
576 entry['package'] = None
577 entry['link_to'] = None
578 if o.inodetype == Target_File.ITYPE_DIRECTORY:
579 entry['isdir'] = 1
580 # is there content in directory
581 entry['childcount'] = Target_File.objects.filter(directory__path=o.path).all().count()
582 else:
583 entry['isdir'] = 0
584
585 # resolve the file to get the package from the resolved file
586 resolved_id = o.sym_target_id
587 resolved_path = o.path
588 if target_packages.count():
589 while resolved_id != "" and resolved_id != None:
590 tf = Target_File.objects.get(pk=resolved_id)
591 resolved_path = tf.path
592 resolved_id = tf.sym_target_id
593
594 thisfile=Package_File.objects.all().filter(path__exact=resolved_path, package_id__in=target_packages)
595 if thisfile.count():
596 p = Package.objects.get(pk=thisfile[0].package_id)
597 entry['installed_package'] = p.installed_name
598 entry['package_id'] = str(p.id)
599 entry['package'] = p.name
600 # don't use resolved path from above, show immediate link-to
601 if o.sym_target_id != "" and o.sym_target_id != None:
602 entry['link_to'] = Target_File.objects.get(pk=o.sym_target_id).path
603 t = template.Template('{% load projecttags %} {{ size|filtered_filesizeformat }}')
604 c = template.Context({'size': o.size})
605 entry['size'] = str(t.render(c))
606 if entry['link_to'] != None:
607 entry['permission'] = node_str[o.inodetype] + o.permission
608 else:
609 entry['permission'] = node_str[o.inodetype] + o.permission
610 entry['owner'] = o.owner
611 entry['group'] = o.group
612 response.append(entry)
613
614 except:
615 pass
616
617 # sort by directories first, then by name
618 rsorted = sorted(response, key=lambda entry : entry['name'])
619 rsorted = sorted(rsorted, key=lambda entry : entry['isdir'], reverse=True)
620 return json.dumps(rsorted, cls=LazyEncoder)
621
622def dirinfo(request, build_id, target_id, file_path=None):
623 template = "dirinfo.html"
624 objects = _get_dir_entries(build_id, target_id, '/')
625 packages_sum = Package.objects.filter(id__in=Target_Installed_Package.objects.filter(target_id=target_id).values('package_id')).aggregate(Sum('installed_size'))
626 dir_list = None
627 if file_path != None:
628 """
629 Link from the included package detail file list page and is
630 requesting opening the dir info to a specific file path.
631 Provide the list of directories to expand and the full path to
632 highlight in the page.
633 """
634 # Aassume target's path separator matches host's, that is, os.sep
635 sep = os.sep
636 dir_list = []
637 head = file_path
638 while head != sep:
639 (head,tail) = os.path.split(head)
640 if head != sep:
641 dir_list.insert(0, head)
642
643 context = { 'build': Build.objects.filter(pk=build_id)[0],
644 'target': Target.objects.filter(pk=target_id)[0],
645 'packages_sum': packages_sum['installed_size__sum'],
646 'objects': objects,
647 'dir_list': dir_list,
648 'file_path': file_path,
649 }
650 return render(request, template, context)
427 651
428def _find_task_dep(task): 652def _find_task_dep(task):
429 tp = [] 653 tp = []
@@ -593,7 +817,7 @@ def tasks_common(request, build_id, variant):
593 } 817 }
594 818
595 } 819 }
596 #if 'tasks' == variant: tc_cache['hidden']='0'; 820 #if 'tasks' == variant: tc_cache['hidden']='0';
597 tc_time={ 821 tc_time={
598 'name':'Time (secs)', 822 'name':'Time (secs)',
599 'qhelp':'How long it took the task to finish, expressed in seconds', 823 'qhelp':'How long it took the task to finish, expressed in seconds',
@@ -796,7 +1020,7 @@ def configvars(request, build_id):
796 # remove duplicate records from multiple search hits in the VariableHistory table 1020 # remove duplicate records from multiple search hits in the VariableHistory table
797 queryset = queryset.distinct() 1021 queryset = queryset.distinct()
798 # remove records where the value is empty AND there are no history files 1022 # remove records where the value is empty AND there are no history files
799 queryset = queryset.exclude(variable_value='',vhistory__file_name__isnull=True) 1023 queryset = queryset.exclude(variable_value='',vhistory__file_name__isnull=True)
800 1024
801 variables = _build_page_range(Paginator(queryset, request.GET.get('count', 50)), request.GET.get('page', 1)) 1025 variables = _build_page_range(Paginator(queryset, request.GET.get('count', 50)), request.GET.get('page', 1))
802 1026
@@ -811,7 +1035,7 @@ def configvars(request, build_id):
811 file_filter += 'conf/distro/' 1035 file_filter += 'conf/distro/'
812 if filter_string.find('/bitbake.conf') > 0: 1036 if filter_string.find('/bitbake.conf') > 0:
813 file_filter += '/bitbake.conf' 1037 file_filter += '/bitbake.conf'
814 1038
815 context = { 1039 context = {
816 'objectname': 'configvars', 1040 'objectname': 'configvars',
817 'object_search_display':'variables', 1041 'object_search_display':'variables',