diff options
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 | ||
196 | class Package(models.Model): | 199 | class 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 @@ | |||
1 | table.treetable span.file { | ||
2 | background-image: url(); | ||
3 | } | ||
4 | |||
5 | table.treetable span.folder { | ||
6 | background-image: url(); | ||
7 | } | ||
8 | |||
9 | table.treetable tr.collapsed span.indenter a { | ||
10 | background-image: url(); | ||
11 | } | ||
12 | |||
13 | table.treetable tr.expanded span.indenter a { | ||
14 | background-image: url(); | ||
15 | } | ||
16 | |||
17 | |||
18 | |||
19 | table.treetable tr.collapsed.selected span.indenter a { | ||
20 | background-image: url(); | ||
21 | } | ||
22 | |||
23 | table.treetable tr.expanded.selected span.indenter a { | ||
24 | background-image: url(); | ||
25 | } | ||
26 | |||
27 | table.treetable tr.accept { | ||
28 | background-color: #a3bce4; | ||
29 | color: #fff | ||
30 | } | ||
31 | |||
32 | table.treetable tr.collapsed.accept td span.indenter a { | ||
33 | background-image: url(); | ||
34 | } | ||
35 | |||
36 | table.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 | /* | ||
2 | table.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 | |||
11 | table.treetable caption { | ||
12 | font-size: .9em; | ||
13 | font-weight: bold; | ||
14 | margin-bottom: .2em; | ||
15 | } | ||
16 | |||
17 | table.treetable tbody tr td { | ||
18 | cursor: default; | ||
19 | padding: .3em 1em; | ||
20 | } | ||
21 | |||
22 | table.treetable span { | ||
23 | background-position: center left; | ||
24 | background-repeat: no-repeat; | ||
25 | padding: .2em 0 .2em 1.5em; | ||
26 | } | ||
27 | */ | ||
28 | |||
29 | table.treetable span.file { | ||
30 | background-image: url(); | ||
31 | } | ||
32 | |||
33 | table.treetable span.folder { | ||
34 | background-image: url(); | ||
35 | } | ||
36 | |||
37 | table.treetable tr.collapsed span.indenter a { | ||
38 | background-image: url(); | ||
39 | } | ||
40 | |||
41 | table.treetable tr.expanded span.indenter a { | ||
42 | background-image: url(); | ||
43 | } | ||
44 | |||
45 | |||
46 | |||
47 | table.treetable tr.collapsed.selected span.indenter a { | ||
48 | background-image: url(); | ||
49 | } | ||
50 | |||
51 | table.treetable tr.expanded.selected span.indenter a { | ||
52 | background-image: url(); | ||
53 | } | ||
54 | |||
55 | table.treetable tr.accept { | ||
56 | background-color: #a3bce4; | ||
57 | color: #fff | ||
58 | } | ||
59 | |||
60 | table.treetable tr.collapsed.accept td span.indenter a { | ||
61 | background-image: url(); | ||
62 | } | ||
63 | |||
64 | table.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 += ' ' + 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 += ' ' + 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 | ||
219 | def 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 | ||
226 | def 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 | ||
234 | def 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 | |||
241 | from django.db import models | ||
242 | from orm.models import Package | ||
243 | @register.filter | ||
244 | def 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 | ||
250 | def 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 | ||
256 | def 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 | ||
22 | import operator | 22 | import operator |
23 | 23 | ||
24 | from django.db.models import Q | 24 | from django.db.models import Q, Sum |
25 | from django.shortcuts import render, redirect | 25 | from django.shortcuts import render, redirect |
26 | from orm.models import Build, Target, Task, Layer, Layer_Version, Recipe, LogMessage, Variable | 26 | from orm.models import Build, Target, Task, Layer, Layer_Version, Recipe, LogMessage, Variable |
27 | from orm.models import Task_Dependency, Recipe_Dependency, Package, Package_File, Package_Dependency | 27 | from orm.models import Task_Dependency, Recipe_Dependency, Package, Package_File, Package_Dependency |
28 | from orm.models import Target_Installed_Package | 28 | from orm.models import Target_Installed_Package, Target_File |
29 | from django.views.decorators.cache import cache_control | 29 | from django.views.decorators.cache import cache_control |
30 | from django.core.paginator import Paginator, EmptyPage, PageNotAnInteger | 30 | from django.core.paginator import Paginator, EmptyPage, PageNotAnInteger |
31 | from django.http import HttpResponseBadRequest | 31 | from django.http import HttpResponseBadRequest |
32 | from django.utils import timezone | 32 | from django.utils import timezone |
33 | from datetime import timedelta | 33 | from datetime import timedelta |
34 | from django.utils import formats | 34 | from django.utils import formats |
35 | import json | ||
35 | 36 | ||
36 | def _build_page_range(paginator, index = 1): | 37 | def _build_page_range(paginator, index = 1): |
37 | try: | 38 | try: |
@@ -163,7 +164,7 @@ def _get_search_results(search_term, queryset, model): | |||
163 | def _search_tuple(request, model): | 164 | def _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 | ||
418 | def target(request, build_id, target_id): | 419 | def 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 | ||
533 | from django.core.serializers.json import DjangoJSONEncoder | ||
534 | from django.http import HttpResponse | ||
535 | def 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 | |||
539 | from django.utils.functional import Promise | ||
540 | from django.utils.encoding import force_text | ||
541 | class 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 | |||
547 | from toastergui.templatetags.projecttags import filtered_filesizeformat | ||
548 | from django import template | ||
549 | import os | ||
550 | def _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 | |||
622 | def 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 | ||
428 | def _find_task_dep(task): | 652 | def _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', |