classes/package: fix FILES_INFO serialisation in pkgdata

The FILES_INFO entry in each pkgdata file stores the list of files for
each package. Make the following improvements to how this is stored:
* Store paths as they would be seen on the target rather than
  erroneously including the full path to PKGDEST (which is specific to
  the build host the package was built on)
* For simplicity when loading the data, store complete paths for each
  entry instead of trying to break off the first part and use it as the
  dict key
* Record sizes for each file (as needed by Toaster)
* Serialise the value explicitly using json rather than just passing it
  through str().

Fixes [YOCTO #5443].

(From OE-Core rev: ca86603607a69a17cc5540d69de0e242b33382d3)

Signed-off-by: Paul Eggleton <paul.eggleton@linux.intel.com>
Signed-off-by: Richard Purdie <richard.purdie@linuxfoundation.org>
This commit is contained in:
Paul Eggleton 2013-12-02 18:50:44 +00:00 committed by Richard Purdie
parent a6450a951f
commit 4a75e83b29
3 changed files with 21 additions and 45 deletions

View File

@ -1110,6 +1110,7 @@ PKGDESTWORK = "${WORKDIR}/pkgdata"
python emit_pkgdata() { python emit_pkgdata() {
from glob import glob from glob import glob
import json
def write_if_exists(f, pkg, var): def write_if_exists(f, pkg, var):
def encode(str): def encode(str):
@ -1173,22 +1174,20 @@ python emit_pkgdata() {
workdir = d.getVar('WORKDIR', True) workdir = d.getVar('WORKDIR', True)
for pkg in packages.split(): for pkg in packages.split():
items = {}
for files_list in pkgfiles[pkg]:
item_name = os.path.basename(files_list)
item_path = os.path.dirname(files_list)
if item_path not in items:
items[item_path] = []
items[item_path].append(item_name)
subdata_file = pkgdatadir + "/runtime/%s" % pkg
pkgval = d.getVar('PKG_%s' % pkg, True) pkgval = d.getVar('PKG_%s' % pkg, True)
if pkgval is None: if pkgval is None:
pkgval = pkg pkgval = pkg
d.setVar('PKG_%s' % pkg, pkg) d.setVar('PKG_%s' % pkg, pkg)
d.setVar('FILES_INFO', str(items)) pkgdestpkg = os.path.join(pkgdest, pkg)
files = {}
for f in pkgfiles[pkg]:
relpth = os.path.relpath(f, pkgdestpkg)
fstat = os.lstat(f)
files[os.sep + relpth] = fstat.st_size
d.setVar('FILES_INFO', json.dumps(files))
subdata_file = pkgdatadir + "/runtime/%s" % pkg
sf = open(subdata_file, 'w') sf = open(subdata_file, 'w')
write_if_exists(sf, pkg, 'PN') write_if_exists(sf, pkg, 'PN')
write_if_exists(sf, pkg, 'PV') write_if_exists(sf, pkg, 'PV')

View File

@ -39,8 +39,7 @@ python toaster_package_dumpdata() {
# scan and send data for each package # scan and send data for each package
import ast import json
import fnmatch
lpkgdata = {} lpkgdata = {}
for pkg in packages.split(): for pkg in packages.split():
@ -54,28 +53,11 @@ python toaster_package_dumpdata() {
(n, v) = line.rstrip().split(":", 1) (n, v) = line.rstrip().split(":", 1)
if pkg in n: if pkg in n:
n = n.replace("_" + pkg, "") n = n.replace("_" + pkg, "")
lpkgdata[n] = v.strip()
line = sf.readline()
pkgsplitname = os.path.join(pkgdest, pkg)
# replace FILES_INFO data with a dictionary of file name - file size
if n == 'FILES_INFO': if n == 'FILES_INFO':
filesizedata = {} lpkgdata[n] = json.loads(v)
val = v.strip().replace('\\\'', '\'') else:
dictval = ast.literal_eval(val) lpkgdata[n] = v.strip()
for parent, dirlist in dictval.items(): line = sf.readline()
idx = parent.find(pkgsplitname)
if idx > -1:
parent = parent[idx+len(pkgsplitname):]
else:
bb.error("Invalid path while looking for file ", parent)
for basename in dirlist:
fullpath = os.path.join(parent, basename)
try:
filesizedata[fullpath] = os.stat(pkgsplitname + fullpath).st_size
except OSError:
# we may hit a symlink that is not pointing correctly over package-split
filesizedata[fullpath] = 0
lpkgdata[n] = filesizedata
# Fire an event containing the pkg data # Fire an event containing the pkg data
bb.event.fire(bb.event.MetadataEvent("SinglePackageInfo", lpkgdata), d) bb.event.fire(bb.event.MetadataEvent("SinglePackageInfo", lpkgdata), d)

View File

@ -262,25 +262,20 @@ def find_path(args, usage, debug=False):
print('ERROR: Unable to find pkgdata directory %s' % pkgdata_dir) print('ERROR: Unable to find pkgdata directory %s' % pkgdata_dir)
sys.exit(1) sys.exit(1)
import ast import json
import fnmatch import fnmatch
for root, dirs, files in os.walk(os.path.join(pkgdata_dir, 'runtime')): for root, dirs, files in os.walk(os.path.join(pkgdata_dir, 'runtime')):
for fn in files: for fn in files:
pkgsplitname = '/packages-split/%s' % fn
with open(os.path.join(root,fn)) as f: with open(os.path.join(root,fn)) as f:
for line in f: for line in f:
if line.startswith('FILES_INFO:'): if line.startswith('FILES_INFO:'):
val = line.split(':', 1)[1].strip().replace('\\\'', '\'') val = line.split(':', 1)[1].strip()
dictval = ast.literal_eval(val) dictval = json.loads(val)
for parent, dirlist in dictval.items(): for fullpth in dictval.keys():
idx = parent.find(pkgsplitname) if fnmatch.fnmatchcase(fullpth, targetpath):
if idx > -1: print("%s: %s" % (fn, fullpth))
parent = parent[idx+len(pkgsplitname):] break
for basename in dirlist:
fullpth = os.path.join(parent, basename)
if fnmatch.fnmatchcase(fullpth, targetpath):
print("%s: %s" % (fn, fullpth))
def main(): def main():