diff options
| author | Paul Eggleton <paul.eggleton@linux.intel.com> | 2011-11-30 16:48:47 +0000 |
|---|---|---|
| committer | Richard Purdie <richard.purdie@linuxfoundation.org> | 2011-12-05 16:23:53 +0000 |
| commit | 6ba6f4a6f744895741c82f93756439292f563bf6 (patch) | |
| tree | 64b35044e9730c94fb12b747cc658d1a2d23ef51 | |
| parent | 45c4f0a5870cf3a089b0edd02f52297865f5b5db (diff) | |
| download | poky-6ba6f4a6f744895741c82f93756439292f563bf6.tar.gz | |
classes/buildhistory: add new output history collection class
Create a new build output history reporting class, using testlab.bbclass
from meta-oe as a base. This records information from images produced by
the build process in text files structured suitably for tracking within
a git repository, thus enabling monitoring of changes over time.
Build history collection can be enabled simply by adding the following
to your local.conf:
INHERIT += "buildhistory"
The output after a build can then be found in BUILDHISTORY_DIR (defaults to
TMPDIR/buildhistory). If you set up this directory as a git repository and
set BUILDHISTORY_COMMIT to "1" in local.conf, the build history data will
be committed on every build.
(From OE-Core rev: 14acb530a27a3b088d0bfd56db291f4e72ace8ab)
Signed-off-by: Paul Eggleton <paul.eggleton@linux.intel.com>
Signed-off-by: Richard Purdie <richard.purdie@linuxfoundation.org>
| -rw-r--r-- | meta/classes/buildhistory.bbclass | 105 | ||||
| -rw-r--r-- | meta/classes/rootfs_ipk.bbclass | 27 | ||||
| -rw-r--r-- | meta/classes/rootfs_rpm.bbclass | 41 |
3 files changed, 167 insertions, 6 deletions
diff --git a/meta/classes/buildhistory.bbclass b/meta/classes/buildhistory.bbclass new file mode 100644 index 0000000000..21f8493ce5 --- /dev/null +++ b/meta/classes/buildhistory.bbclass | |||
| @@ -0,0 +1,105 @@ | |||
| 1 | # | ||
| 2 | # Records history of build output in order to detect regressions | ||
| 3 | # | ||
| 4 | # Based in part on testlab.bbclass | ||
| 5 | # | ||
| 6 | # Copyright (C) 2011 Intel Corporation | ||
| 7 | # Copyright (C) 2007-2011 Koen Kooi <koen@openembedded.org> | ||
| 8 | # | ||
| 9 | |||
| 10 | BUILDHISTORY_DIR ?= "${TMPDIR}/buildhistory" | ||
| 11 | BUILDHISTORY_DIR_IMAGE = "${BUILDHISTORY_DIR}/images/${MACHINE_ARCH}/${TCLIBC}/${IMAGE_BASENAME}" | ||
| 12 | BUILDHISTORY_COMMIT ?= "0" | ||
| 13 | BUILDHISTORY_COMMIT_AUTHOR ?= "buildhistory <buildhistory@${DISTRO}>" | ||
| 14 | BUILDHISTORY_PUSH_REPO ?= "" | ||
| 15 | |||
| 16 | buildhistory_get_image_installed() { | ||
| 17 | # Anything requiring the use of the packaging system should be done in here | ||
| 18 | # in case the packaging files are going to be removed for this image | ||
| 19 | |||
| 20 | mkdir -p ${BUILDHISTORY_DIR_IMAGE} | ||
| 21 | |||
| 22 | # Get list of installed packages | ||
| 23 | list_installed_packages | sort > ${BUILDHISTORY_DIR_IMAGE}/installed-package-names.txt | ||
| 24 | INSTALLED_PKGS=`cat ${BUILDHISTORY_DIR_IMAGE}/installed-package-names.txt` | ||
| 25 | |||
| 26 | # Produce installed package file and size lists and dependency graph | ||
| 27 | echo -n > ${BUILDHISTORY_DIR_IMAGE}/installed-packages.txt | ||
| 28 | echo -n > ${BUILDHISTORY_DIR_IMAGE}/installed-package-sizes.tmp | ||
| 29 | echo -e "digraph depends {\n node [shape=plaintext]" > ${BUILDHISTORY_DIR_IMAGE}/depends.dot | ||
| 30 | for pkg in $INSTALLED_PKGS; do | ||
| 31 | pkgfile=`get_package_filename $pkg` | ||
| 32 | echo `basename $pkgfile` >> ${BUILDHISTORY_DIR_IMAGE}/installed-packages.txt | ||
| 33 | if [ -f $pkgfile ] ; then | ||
| 34 | pkgsize=`du -k $pkgfile | head -n1 | awk '{ print $1 }'` | ||
| 35 | echo $pkgsize $pkg >> ${BUILDHISTORY_DIR_IMAGE}/installed-package-sizes.tmp | ||
| 36 | fi | ||
| 37 | |||
| 38 | deps=`list_package_depends $pkg` | ||
| 39 | for dep in $deps ; do | ||
| 40 | echo "$pkg OPP $dep;" | sed -e 's:-:_:g' -e 's:\.:_:g' -e 's:+::g' | sed 's:OPP:->:g' >> ${BUILDHISTORY_DIR_IMAGE}/depends.dot | ||
| 41 | done | ||
| 42 | |||
| 43 | recs=`list_package_recommends $pkg` | ||
| 44 | for rec in $recs ; do | ||
| 45 | echo "$pkg OPP $rec [style=dotted];" | sed -e 's:-:_:g' -e 's:\.:_:g' -e 's:+::g' | sed 's:OPP:->:g' >> ${BUILDHISTORY_DIR_IMAGE}/depends.dot | ||
| 46 | done | ||
| 47 | done | ||
| 48 | echo "}" >> ${BUILDHISTORY_DIR_IMAGE}/depends.dot | ||
| 49 | |||
| 50 | cat ${BUILDHISTORY_DIR_IMAGE}/installed-package-sizes.tmp | sort -n -r | awk '{print $1 "\tKiB " $2}' > ${BUILDHISTORY_DIR_IMAGE}/installed-package-sizes.txt | ||
| 51 | rm ${BUILDHISTORY_DIR_IMAGE}/installed-package-sizes.tmp | ||
| 52 | |||
| 53 | # Produce some cut-down graphs (for readability) | ||
| 54 | grep -v kernel_image ${BUILDHISTORY_DIR_IMAGE}/depends.dot | grep -v kernel_2 | grep -v kernel_3 > ${BUILDHISTORY_DIR_IMAGE}/depends-nokernel.dot | ||
| 55 | grep -v libc6 ${BUILDHISTORY_DIR_IMAGE}/depends-nokernel.dot | grep -v libgcc > ${BUILDHISTORY_DIR_IMAGE}/depends-nokernel-nolibc.dot | ||
| 56 | grep -v update_ ${BUILDHISTORY_DIR_IMAGE}/depends-nokernel-nolibc.dot > ${BUILDHISTORY_DIR_IMAGE}/depends-nokernel-nolibc-noupdate.dot | ||
| 57 | grep -v kernel_module ${BUILDHISTORY_DIR_IMAGE}/depends-nokernel-nolibc-noupdate.dot > ${BUILDHISTORY_DIR_IMAGE}/depends-nokernel-nolibc-noupdate-nomodules.dot | ||
| 58 | |||
| 59 | # Workaround for broken shell function dependencies | ||
| 60 | if false ; then | ||
| 61 | get_package_filename | ||
| 62 | list_package_depends | ||
| 63 | list_package_recommends | ||
| 64 | fi | ||
| 65 | } | ||
| 66 | |||
| 67 | buildhistory_get_imageinfo() { | ||
| 68 | # List the files in the image, but exclude date/time etc. | ||
| 69 | # This awk script is somewhat messy, but handles where the size is not printed for device files under pseudo | ||
| 70 | find ${IMAGE_ROOTFS} -ls | awk '{ if ( $7 ~ /[0-9]/ ) printf "%s %10-s %10-s %10s %s %s %s\n", $3, $5, $6, $7, $11, $12, $13 ; else printf "%s %10-s %10-s %10s %s %s %s\n", $3, $5, $6, 0, $10, $11, $12 }' > ${BUILDHISTORY_DIR_IMAGE}/files-in-image.txt | ||
| 71 | |||
| 72 | # Add some configuration information | ||
| 73 | echo "${MACHINE}: ${IMAGE_BASENAME} configured for ${DISTRO} ${DISTRO_VERSION}" > ${BUILDHISTORY_DIR_IMAGE}/build-id | ||
| 74 | echo "${@buildhistory_get_layers(d)}" >> ${BUILDHISTORY_DIR_IMAGE}/build-id | ||
| 75 | } | ||
| 76 | |||
| 77 | # By prepending we get in before the removal of packaging files | ||
| 78 | ROOTFS_POSTPROCESS_COMMAND =+ "buildhistory_get_image_installed ; " | ||
| 79 | |||
| 80 | IMAGE_POSTPROCESS_COMMAND += " buildhistory_get_imageinfo ; " | ||
| 81 | |||
| 82 | def buildhistory_get_layers(d): | ||
| 83 | layertext = "Configured metadata layers:\n%s\n" % '\n'.join(get_layers_branch_rev(d)) | ||
| 84 | return layertext | ||
| 85 | |||
| 86 | |||
| 87 | buildhistory_commit() { | ||
| 88 | ( cd ${BUILDHISTORY_DIR}/ | ||
| 89 | git add ${BUILDHISTORY_DIR}/* | ||
| 90 | git commit ${BUILDHISTORY_DIR}/ -m "Build ${BUILDNAME} for machine ${MACHINE} configured for ${DISTRO} ${DISTRO_VERSION}" --author "${BUILDHISTORY_COMMIT_AUTHOR}" > /dev/null | ||
| 91 | if [ "${BUILDHISTORY_PUSH_REPO}" != "" ] ; then | ||
| 92 | git push -q ${BUILDHISTORY_PUSH_REPO} | ||
| 93 | fi) || true | ||
| 94 | } | ||
| 95 | |||
| 96 | python buildhistory_eventhandler() { | ||
| 97 | import bb.build | ||
| 98 | import bb.event | ||
| 99 | |||
| 100 | if isinstance(e, bb.event.BuildCompleted): | ||
| 101 | if e.data.getVar("BUILDHISTORY_COMMIT", True) == "1": | ||
| 102 | bb.build.exec_func("buildhistory_commit", e.data) | ||
| 103 | } | ||
| 104 | |||
| 105 | addhandler buildhistory_eventhandler | ||
diff --git a/meta/classes/rootfs_ipk.bbclass b/meta/classes/rootfs_ipk.bbclass index 4a5a2dd3be..b4b95c5645 100644 --- a/meta/classes/rootfs_ipk.bbclass +++ b/meta/classes/rootfs_ipk.bbclass | |||
| @@ -143,11 +143,36 @@ remove_packaging_data_files() { | |||
| 143 | mkdir ${IMAGE_ROOTFS}${opkglibdir} | 143 | mkdir ${IMAGE_ROOTFS}${opkglibdir} |
| 144 | } | 144 | } |
| 145 | 145 | ||
| 146 | list_installed_packages() { | ||
| 147 | grep ^Package: ${IMAGE_ROOTFS}${opkglibdir}/status | sed "s/^Package: //" | ||
| 148 | } | ||
| 149 | |||
| 150 | get_package_filename() { | ||
| 151 | name=`opkg-cl ${IPKG_ARGS} info $1 | grep -B 7 -A 7 "^Status.* \(\(installed\)\|\(unpacked\)\)" | awk '/^Package/ {printf $2"_"}'` | ||
| 152 | name=$name`opkg-cl ${IPKG_ARGS} info $1 | grep -B 7 -A 7 "^Status.* \(\(installed\)\|\(unpacked\)\)" | awk -F: '/^Version/ {printf $NF"_"}' | sed 's/^\s*//g'` | ||
| 153 | name=$name`opkg-cl ${IPKG_ARGS} info $1 | grep -B 7 -A 7 "^Status.* \(\(installed\)\|\(unpacked\)\)" | awk '/^Archi/ {print $2".ipk"}'` | ||
| 154 | |||
| 155 | fullname=`find ${DEPLOY_DIR_IPK} -name "$name" || true` | ||
| 156 | if [ "$fullname" = "" ] ; then | ||
| 157 | echo $name | ||
| 158 | else | ||
| 159 | echo $fullname | ||
| 160 | fi | ||
| 161 | } | ||
| 162 | |||
| 163 | list_package_depends() { | ||
| 164 | opkg-cl ${IPKG_ARGS} info $1 | grep ^Depends | sed -e 's/^Depends: //' -e 's/,//g' -e 's:([=<>]* [0-9a-zA-Z.~\-]*)::g' | ||
| 165 | } | ||
| 166 | |||
| 167 | list_package_recommends() { | ||
| 168 | opkg-cl ${IPKG_ARGS} info $1 | grep ^Recommends | sed -e 's/^Recommends: //' -e 's/,//g' -e 's:([=<>]* [0-9a-zA-Z.~\-]*)::g' | ||
| 169 | } | ||
| 170 | |||
| 146 | install_all_locales() { | 171 | install_all_locales() { |
| 147 | 172 | ||
| 148 | PACKAGES_TO_INSTALL="" | 173 | PACKAGES_TO_INSTALL="" |
| 149 | 174 | ||
| 150 | INSTALLED_PACKAGES=`grep ^Package: ${IMAGE_ROOTFS}${opkglibdir}/status |sed "s/^Package: //"|egrep -v -- "(-locale-|-dev$|-doc$|^kernel|^glibc|^ttf|^task|^perl|^python)"` | 175 | INSTALLED_PACKAGES=`list_installed_packages | egrep -v -- "(-locale-|-dev$|-doc$|^kernel|^glibc|^ttf|^task|^perl|^python)"` |
| 151 | 176 | ||
| 152 | for pkg in $INSTALLED_PACKAGES | 177 | for pkg in $INSTALLED_PACKAGES |
| 153 | do | 178 | do |
diff --git a/meta/classes/rootfs_rpm.bbclass b/meta/classes/rootfs_rpm.bbclass index 6973008c59..5fd45d758c 100644 --- a/meta/classes/rootfs_rpm.bbclass +++ b/meta/classes/rootfs_rpm.bbclass | |||
| @@ -160,16 +160,47 @@ remove_packaging_data_files() { | |||
| 160 | rm -rf ${IMAGE_ROOTFS}${opkglibdir} | 160 | rm -rf ${IMAGE_ROOTFS}${opkglibdir} |
| 161 | } | 161 | } |
| 162 | 162 | ||
| 163 | RPM_QUERY_CMD = '${RPM} --root ${IMAGE_ROOTFS} -D "_dbpath ${rpmlibdir}" \ | ||
| 164 | -D "__dbi_txn create nofsync private"' | ||
| 165 | |||
| 166 | list_installed_packages() { | ||
| 167 | ${RPM_QUERY_CMD} -qa --qf "[%{NAME}\n]" | ||
| 168 | } | ||
| 169 | |||
| 170 | get_package_filename() { | ||
| 171 | resolve_package_rpm ${RPMCONF_TARGET_BASE}-base_archs.conf $1 | ||
| 172 | } | ||
| 173 | |||
| 174 | list_package_depends() { | ||
| 175 | pkglist=`list_installed_packages` | ||
| 176 | |||
| 177 | for req in `${RPM_QUERY_CMD} -q --qf "[%{REQUIRES}\n]" $1`; do | ||
| 178 | if echo "$req" | grep -q "^rpmlib" ; then continue ; fi | ||
| 179 | |||
| 180 | realpkg="" | ||
| 181 | for dep in $pkglist; do | ||
| 182 | if [ "$dep" = "$req" ] ; then | ||
| 183 | realpkg="1" | ||
| 184 | echo $req | ||
| 185 | break | ||
| 186 | fi | ||
| 187 | done | ||
| 188 | |||
| 189 | if [ "$realdep" = "" ] ; then | ||
| 190 | ${RPM_QUERY_CMD} -q --whatprovides $req --qf "%{NAME}\n" | ||
| 191 | fi | ||
| 192 | done | ||
| 193 | } | ||
| 194 | |||
| 195 | list_package_recommends() { | ||
| 196 | : | ||
| 197 | } | ||
| 163 | 198 | ||
| 164 | install_all_locales() { | 199 | install_all_locales() { |
| 165 | PACKAGES_TO_INSTALL="" | 200 | PACKAGES_TO_INSTALL="" |
| 166 | 201 | ||
| 167 | # Generate list of installed packages... | 202 | # Generate list of installed packages... |
| 168 | INSTALLED_PACKAGES=$( \ | 203 | INSTALLED_PACKAGES=`list_installed_packages | egrep -v -- "(-locale-|-dev$|-doc$|^kernel|^glibc|^ttf|^task|^perl|^python)"` |
| 169 | ${RPM} --root ${IMAGE_ROOTFS} -D "_dbpath ${rpmlibdir}" \ | ||
| 170 | -D "__dbi_txn create nofsync private" \ | ||
| 171 | -qa --qf "[%{NAME}\n]" | egrep -v -- "(-locale-|-dev$|-doc$|^kernel|^glibc|^ttf|^task|^perl|^python)" \ | ||
| 172 | ) | ||
| 173 | 204 | ||
| 174 | # This would likely be faster if we did it in one transaction | 205 | # This would likely be faster if we did it in one transaction |
| 175 | # but this should be good enough for the few users of this function... | 206 | # but this should be good enough for the few users of this function... |
