Split out postprocess+commit to separate script
As this logic is of most interest to potential other consumers; it's not entangled with yum for example.
This commit is contained in:
parent
df1db5f85b
commit
7333091db4
@ -15,14 +15,20 @@
|
||||
# Free Software Foundation, Inc., 59 Temple Place - Suite 330,
|
||||
# Boston, MA 02111-1307, USA.
|
||||
|
||||
bin_SCRIPTS += rpm-ostree
|
||||
bin_SCRIPTS += rpm-ostree rpm-ostree-postprocess-and-commit
|
||||
|
||||
rpm-ostree: src/rpm-ostree.in Makefile
|
||||
$(AM_V_GEN) sed -e s,@PYTHON\@,$(PYTHON), -e s,@pkglibdir\@,$(pkglibdir), $< > $@.tmp && mv $@.tmp $@
|
||||
@chmod a+x $@
|
||||
|
||||
rpm-ostree-postprocess-and-commit: src/rpm-ostree-postprocess-and-commit.in Makefile
|
||||
$(AM_V_GEN) sed -e s,@PYTHON\@,$(PYTHON), -e s,@pkglibdir\@,$(pkglibdir), $< > $@.tmp && mv $@.tmp $@
|
||||
@chmod a+x $@
|
||||
|
||||
pyrpmostreedir = $(pkglibdir)/rpm-ostree
|
||||
pyrpmostree_DATA = src/rpmostree.py \
|
||||
pyrpmostree_DATA = \
|
||||
src/rpmostree.py \
|
||||
src/rpmostreepost.py \
|
||||
$(NULL)
|
||||
|
||||
privdatadir=$(pkglibdir)
|
||||
|
@ -20,7 +20,12 @@
|
||||
{
|
||||
"docker-io": { "packages": ["@core", "@standard", "docker-io"] },
|
||||
"freeipa-server": { "packages": ["@core", "@standard", "docker-io"] },
|
||||
"jbossas": { "packages": ["@core", "@standard", "@jbossas"] }
|
||||
"jbossas": { "packages": ["@core", "@standard", "@jbossas"] },
|
||||
"virthost": { "packages": ["@core", "@standard",
|
||||
"libvirt-daemon-kvm",
|
||||
"libvirt-daemon-config-network",
|
||||
"qemu-kvm",
|
||||
"libguestfs-tools"] }
|
||||
},
|
||||
"workstation":
|
||||
{
|
||||
|
@ -36,6 +36,7 @@ make install DESTDIR=$RPM_BUILD_ROOT INSTALL="install -p -c"
|
||||
%files
|
||||
%doc COPYING README.md
|
||||
%{_bindir}/rpm-ostree
|
||||
%{_bindir}/rpm-ostree-postprocess-and-commit
|
||||
%{_bindir}/rpm-ostree-autobuilder
|
||||
%{_bindir}/fedostree-make-trees
|
||||
%{_bindir}/fedostree-make-trees-loop
|
||||
|
@ -18,6 +18,8 @@
|
||||
const GLib = imports.gi.GLib;
|
||||
const Gio = imports.gi.Gio;
|
||||
|
||||
const GSystem = imports.gi.GSystem;
|
||||
|
||||
const Params = imports.params;
|
||||
|
||||
function walkDirInternal(dir, matchParams, callback, cancellable, queryStr, depth, sortByName) {
|
||||
@ -87,3 +89,10 @@ function walkDir(dir, matchParams, callback, cancellable) {
|
||||
let depth = matchParams.depth;
|
||||
walkDirInternal(dir, matchParams, callback, cancellable, queryStr, depth, matchParams.sortByName);
|
||||
}
|
||||
|
||||
function openReplace(path, cancellable) {
|
||||
GSystem.shutil_rm_rf(path, cancellable);
|
||||
return path.replace(null, false,
|
||||
Gio.FileCreateFlags.REPLACE_DESTINATION,
|
||||
cancellable);
|
||||
}
|
||||
|
@ -44,28 +44,40 @@ const TaskBuild = new Lang.Class({
|
||||
|
||||
DefaultParameters: {forceComponents: []},
|
||||
|
||||
_composeProduct: function(productName, treeName, treeData, release, architecture, cancellable) {
|
||||
_composeProduct: function(ref, productName, treeName, treeData, release, architecture, cancellable) {
|
||||
let repos = ['fedora-' + release,
|
||||
'walters-nss-altfiles'];
|
||||
if (release != 'rawhide')
|
||||
repos.push('fedora-' + release + '-updates');
|
||||
|
||||
let ref = [this._productData['osname'], release, architecture, productName].join('/');
|
||||
let packages = treeData['packages'];
|
||||
let baseRequired = this._productData['base_required_packages'];
|
||||
print("packages=" + JSON.stringify(packages) + " baseRequired=" + JSON.stringify(baseRequired));
|
||||
packages.push.apply(packages, baseRequired);
|
||||
|
||||
print("Starting build of " + ref);
|
||||
|
||||
let argv = ['rpm-ostree',
|
||||
'--repo=' + this.workdir.get_child('repo').get_path()];
|
||||
argv.push.apply(argv, repos.map(function (a) { return '--enablerepo=' + a; }));
|
||||
argv.push.apply(argv, ['--os=fedora', '--os-version=' + release,
|
||||
'create', ref]);
|
||||
argv.push.apply(argv, packages);
|
||||
let productNameUnix = productName.replace(/\//g, '_');
|
||||
let productNameUnix = ref.replace(/\//g, '_');
|
||||
let buildOutputPath = Gio.File.new_for_path('log-' + productNameUnix + '.txt');
|
||||
ProcUtil.runSync(argv, cancellable, { logInitiation: true,
|
||||
cwd: this.workdir });
|
||||
|
||||
let procContext = new GSystem.SubprocessContext({ argv: argv });
|
||||
GSystem.shutil_rm_rf(buildOutputPath, cancellable);
|
||||
procContext.set_stdout_file_path(buildOutputPath.get_path());
|
||||
procContext.set_stderr_disposition(GSystem.SubprocessStreamDisposition.STDERR_MERGE);
|
||||
let proc = new GSystem.Subprocess({ context: procContext });
|
||||
proc.init(cancellable);
|
||||
try {
|
||||
proc.wait_sync_check(cancellable);
|
||||
} catch (e) {
|
||||
print("Build of " + productName + " failed");
|
||||
return false;
|
||||
}
|
||||
return true;
|
||||
},
|
||||
|
||||
execute: function(cancellable) {
|
||||
@ -76,16 +88,26 @@ const TaskBuild = new Lang.Class({
|
||||
let releases = productData['releases'];
|
||||
let architectures = productData['architectures'];
|
||||
let products = productData['products'];
|
||||
let successful = [];
|
||||
let failed = [];
|
||||
for (let i = 0; i < releases.length; i++) {
|
||||
for (let j = 0; j < architectures.length; j++) {
|
||||
for (let productName in products) {
|
||||
for (let treeName in products[productName]) {
|
||||
this._composeProduct(productName, treeName, products[productName][treeName],
|
||||
releases[i], architectures[j],
|
||||
cancellable);
|
||||
let release = releases[i];
|
||||
let architecture = architectures[j];
|
||||
let ref = [this._productData['osname'], release, architecture, productName, treeName].join('/');
|
||||
if (this._composeProduct(ref, productName, treeName, products[productName][treeName],
|
||||
release, architecture,
|
||||
cancellable))
|
||||
successful.push(ref);
|
||||
else
|
||||
failed.push(ref);
|
||||
}
|
||||
}
|
||||
}
|
||||
}
|
||||
print("Successful: " + successful.join(' '));
|
||||
print("Failed: " + failed.join(' '));
|
||||
}
|
||||
});
|
||||
|
31
src/rpm-ostree-postprocess-and-commit.in
Normal file
31
src/rpm-ostree-postprocess-and-commit.in
Normal file
@ -0,0 +1,31 @@
|
||||
#!@PYTHON@
|
||||
#
|
||||
# Copyright (C) 2013 Colin Walters <walters@verbum.org>
|
||||
#
|
||||
# This library is free software; you can redistribute it and/or
|
||||
# modify it under the terms of the GNU Lesser General Public
|
||||
# License as published by the Free Software Foundation; either
|
||||
# version 2 of the License, or (at your option) any later version.
|
||||
#
|
||||
# This library is distributed in the hope that it will be useful,
|
||||
# but WITHOUT ANY WARRANTY; without even the implied warranty of
|
||||
# MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
|
||||
# Lesser General Public License for more details.
|
||||
#
|
||||
# You should have received a copy of the GNU Lesser General Public
|
||||
# License along with this library; if not, write to the
|
||||
# Free Software Foundation, Inc., 59 Temple Place - Suite 330,
|
||||
# Boston, MA 02111-1307, USA.
|
||||
|
||||
import os
|
||||
import sys
|
||||
import __builtin__
|
||||
|
||||
__builtin__.__dict__['PKGLIBDIR'] = "@pkglibdir@"
|
||||
|
||||
path = os.path.join('@pkglibdir@', 'rpm-ostree')
|
||||
sys.path.insert(0, path)
|
||||
|
||||
import rpmostreepost
|
||||
|
||||
sys.exit(rpmostreepost.main())
|
158
src/rpmostree.py
158
src/rpmostree.py
@ -27,7 +27,6 @@ import subprocess
|
||||
|
||||
from gi.repository import GLib
|
||||
from gi.repository import Gio
|
||||
from gi.repository import OSTree
|
||||
|
||||
os_release_data = {}
|
||||
opts = None
|
||||
@ -60,36 +59,6 @@ def _find_current_origin_refspec():
|
||||
return line[len('refspec='):]
|
||||
return None
|
||||
|
||||
def _initfs(targetroot):
|
||||
os.makedirs(targetroot)
|
||||
for d in ['dev', 'proc', 'run', 'sys', 'var']:
|
||||
os.mkdir(os.path.join(targetroot, d))
|
||||
|
||||
# Special ostree mount
|
||||
os.mkdir(os.path.join(targetroot, 'sysroot'))
|
||||
|
||||
# Some FHS targets; these all live in /var
|
||||
for (target, name) in [('var/opt', 'opt'),
|
||||
('var/srv', 'srv'),
|
||||
('var/mnt', 'mnt'),
|
||||
('var/roothome', 'root'),
|
||||
('var/home', 'home'),
|
||||
('run/media', 'media'),
|
||||
('sysroot/ostree', 'ostree'),
|
||||
('sysroot/tmp', 'tmp')]:
|
||||
os.symlink(target, os.path.join(targetroot, name))
|
||||
|
||||
|
||||
def _clone_current_root_to_yumroot(yumroot):
|
||||
_initfs(yumroot)
|
||||
for d in ['boot', 'usr', 'etc', 'var/lib/rpm', 'var/cache/yum']:
|
||||
destdir = os.path.join(yumroot, d)
|
||||
rmrf(destdir)
|
||||
ensuredir(os.path.dirname(destdir))
|
||||
subprocess.check_call(['cp', '--reflink=auto', '-a',
|
||||
'/' + d,
|
||||
destdir])
|
||||
|
||||
def replace_nsswitch(target_usretc):
|
||||
nsswitch_conf = os.path.join(target_usretc, 'nsswitch.conf')
|
||||
f = open(nsswitch_conf)
|
||||
@ -164,64 +133,6 @@ def do_kernel_prep(yumroot, logs_lookaside):
|
||||
if os.path.exists(varlog_dracut_path):
|
||||
os.rename(varlog_dracut_path, os.path.join(logs_lookaside, 'dracut.log'))
|
||||
|
||||
def _create_rootfs_from_yumroot_content(targetroot, yumroot):
|
||||
"""Prepare a root filesystem, taking mainly the contents of /usr from yumroot"""
|
||||
|
||||
_initfs(targetroot)
|
||||
|
||||
# We take /usr from the yum content
|
||||
os.rename(os.path.join(yumroot, 'usr'), os.path.join(targetroot, 'usr'))
|
||||
# Plus the RPM database goes in usr/share/rpm
|
||||
legacyrpm_path = os.path.join(yumroot, 'var/lib/rpm')
|
||||
newrpm_path = os.path.join(targetroot, 'usr/share/rpm')
|
||||
if not os.path.isdir(newrpm_path):
|
||||
os.rename(legacyrpm_path, newrpm_path)
|
||||
|
||||
# Except /usr/local -> ../var/usrlocal
|
||||
target_usrlocal = os.path.join(targetroot, 'usr/local')
|
||||
if not os.path.islink(target_usrlocal):
|
||||
rmrf(target_usrlocal)
|
||||
os.symlink('../var/usrlocal', target_usrlocal)
|
||||
target_usretc = os.path.join(targetroot, 'usr/etc')
|
||||
rmrf(target_usretc)
|
||||
os.rename(os.path.join(yumroot, 'etc'), target_usretc)
|
||||
|
||||
# Move boot, but rename the kernel/initramfs to have a checksum
|
||||
targetboot = os.path.join(targetroot, 'boot')
|
||||
os.rename(os.path.join(yumroot, 'boot'), targetboot)
|
||||
kernel = None
|
||||
initramfs = None
|
||||
for name in os.listdir(targetboot):
|
||||
if name.startswith('vmlinuz-'):
|
||||
kernel = os.path.join(targetboot, name)
|
||||
elif name.startswith('initramfs-'):
|
||||
initramfs = os.path.join(targetboot, name)
|
||||
|
||||
assert (kernel is not None and initramfs is not None)
|
||||
|
||||
checksum = GLib.Checksum.new(GLib.ChecksumType.SHA256)
|
||||
f = open(kernel)
|
||||
feed_checksum(checksum, f)
|
||||
f.close()
|
||||
f = open(initramfs)
|
||||
feed_checksum(checksum, f)
|
||||
f.close()
|
||||
|
||||
bootcsum = checksum.get_string()
|
||||
|
||||
os.rename(kernel, kernel + '-' + bootcsum)
|
||||
os.rename(initramfs, initramfs + '-' + bootcsum)
|
||||
|
||||
# Also carry along toplevel compat links
|
||||
for name in ['lib', 'lib64', 'bin', 'sbin']:
|
||||
src = os.path.join(yumroot, name)
|
||||
if os.path.islink(src):
|
||||
os.rename(src, os.path.join(targetroot, name))
|
||||
|
||||
target_tmpfilesd = os.path.join(targetroot, 'usr/lib/tmpfiles.d')
|
||||
ensuredir(target_tmpfilesd)
|
||||
shutil.copy(os.path.join(PKGLIBDIR, 'tmpfiles-ostree-integration.conf'), target_tmpfilesd)
|
||||
|
||||
def runyum(argv, yumroot, stdin_str=None):
|
||||
yumargs = list(['yum', '-y', '--releasever=%s' % (opts.os_version, ), '--nogpg', '--setopt=keepcache=1', '--installroot=' + yumroot, '--disablerepo=*'])
|
||||
yumargs.extend(map(lambda x: '--enablerepo=' + x, opts.enablerepo))
|
||||
@ -273,10 +184,6 @@ def main():
|
||||
action='store',
|
||||
default=None,
|
||||
help="Stop at given phase")
|
||||
parser.add_option('', "--gpg-sign",
|
||||
action='store',
|
||||
default=None,
|
||||
help="Sign commits using given GPG key ID")
|
||||
parser.add_option('', "--name",
|
||||
action='store',
|
||||
default=None,
|
||||
@ -328,13 +235,6 @@ def main():
|
||||
print >>sys.stderr, "Unknown action %s" % (action, )
|
||||
sys.exit(1)
|
||||
|
||||
if opts.repo_path is not None:
|
||||
repo = OSTree.Repo.new(Gio.File.new_for_path(opts.repo_path))
|
||||
else:
|
||||
repo = OSTree.Repo.new_default()
|
||||
|
||||
repo.open(None)
|
||||
|
||||
cachedir = '/var/cache/rpm-ostree/work'
|
||||
ensuredir(cachedir)
|
||||
|
||||
@ -348,19 +248,13 @@ def main():
|
||||
ensuredir(logs_lookaside)
|
||||
|
||||
shutil.rmtree(yumroot, ignore_errors=True)
|
||||
if action == 'create':
|
||||
yumroot_varcache = os.path.join(yumroot, 'var/cache')
|
||||
if os.path.isdir(yumcache_lookaside):
|
||||
log("Reusing cache: " + yumroot_varcache)
|
||||
ensuredir(yumroot_varcache)
|
||||
subprocess.check_call(['cp', '-a', yumcache_lookaside, yumcachedir])
|
||||
else:
|
||||
log("No cache found at: " + yumroot_varcache)
|
||||
yumroot_varcache = os.path.join(yumroot, 'var/cache')
|
||||
if os.path.isdir(yumcache_lookaside):
|
||||
log("Reusing cache: " + yumroot_varcache)
|
||||
ensuredir(yumroot_varcache)
|
||||
subprocess.check_call(['cp', '-a', yumcache_lookaside, yumcachedir])
|
||||
else:
|
||||
log("Cloning active root")
|
||||
_clone_current_root_to_yumroot(yumroot)
|
||||
log("...done")
|
||||
time.sleep(3)
|
||||
log("No cache found at: " + yumroot_varcache)
|
||||
|
||||
# Ensure we have enough to modify NSS
|
||||
yuminstall(yumroot, ['filesystem', 'glibc', 'nss-altfiles', 'shadow-utils'])
|
||||
@ -394,36 +288,12 @@ def main():
|
||||
manifest = subprocess.check_call(['rpm', '-qa', '--dbpath=' + yumroot_rpmlibdir],
|
||||
stdout=open(rpmtextlist, 'w'))
|
||||
|
||||
rmrf(targetroot)
|
||||
_create_rootfs_from_yumroot_content(targetroot, yumroot)
|
||||
argv = ['rpm-ostree-postprocess-and-commit',
|
||||
'--repo=' + opts.repo_path,
|
||||
'-m', commit_message,
|
||||
yumroot,
|
||||
ref]
|
||||
log("Running: %s" % (subprocess.list2cmdline(argv), ))
|
||||
subprocess.check_call(argv)
|
||||
|
||||
yumroot_varlog = os.path.join(yumroot, 'var/log')
|
||||
for name in os.listdir(yumroot_varlog):
|
||||
shutil.move(os.path.join(yumroot_varlog, name), logs_lookaside)
|
||||
|
||||
# To make SELinux work, we need to do the labeling right before this.
|
||||
# This really needs some sort of API, so we can apply the xattrs as
|
||||
# we're committing into the repo, rather than having to label the
|
||||
# physical FS.
|
||||
# For now, no xattrs (and hence no SELinux =( )
|
||||
log("Committing " + targetroot + "...")
|
||||
repo.prepare_transaction(None)
|
||||
mtree = OSTree.MutableTree.new()
|
||||
modifier = OSTree.RepoCommitModifier.new(OSTree.RepoCommitModifierFlags.SKIP_XATTRS, None, None)
|
||||
repo.write_directory_to_mtree(Gio.File.new_for_path(targetroot),
|
||||
mtree, modifier, None)
|
||||
[success, parent] = repo.resolve_rev(ref, True)
|
||||
[success, tree] = repo.write_mtree(mtree, None)
|
||||
[success, commit] = repo.write_commit(parent, '', commit_message, None, tree, None)
|
||||
if opts.gpg_sign is not None:
|
||||
repo.sign_commit(commit, opts.gpg_sign, None, None)
|
||||
repo.transaction_set_ref(None, ref, commit)
|
||||
repo.commit_transaction(None)
|
||||
|
||||
log("%s => %s" % (ref, commit))
|
||||
|
||||
rmrf(yumroot)
|
||||
rmrf(targetroot)
|
||||
|
||||
if opts.deploy:
|
||||
subprocess.check_call(['ostree', 'admin', 'deploy', '--os=' + opts.os, ref])
|
||||
log("Complete")
|
||||
|
277
src/rpmostreepost.py
Normal file
277
src/rpmostreepost.py
Normal file
@ -0,0 +1,277 @@
|
||||
#!/usr/bin/env python
|
||||
#
|
||||
# Copyright (C) 2012,2013,2014 Colin Walters <walters@verbum.org>
|
||||
#
|
||||
# This library is free software; you can redistribute it and/or
|
||||
# modify it under the terms of the GNU Lesser General Public
|
||||
# License as published by the Free Software Foundation; either
|
||||
# version 2 of the License, or (at your option) any later version.
|
||||
#
|
||||
# This library is distributed in the hope that it will be useful,
|
||||
# but WITHOUT ANY WARRANTY; without even the implied warranty of
|
||||
# MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
|
||||
# Lesser General Public License for more details.
|
||||
#
|
||||
# You should have received a copy of the GNU Lesser General Public
|
||||
# License along with this library; if not, write to the
|
||||
# Free Software Foundation, Inc., 59 Temple Place - Suite 330,
|
||||
# Boston, MA 02111-1307, USA.
|
||||
|
||||
import os
|
||||
import re
|
||||
import sys
|
||||
import optparse
|
||||
import time
|
||||
import shutil
|
||||
import subprocess
|
||||
|
||||
from gi.repository import GLib
|
||||
from gi.repository import Gio
|
||||
from gi.repository import OSTree
|
||||
|
||||
def log(msg):
|
||||
sys.stdout.write(msg)
|
||||
sys.stdout.write('\n')
|
||||
sys.stdout.flush()
|
||||
|
||||
def ensuredir(path):
|
||||
if not os.path.isdir(path):
|
||||
os.makedirs(path)
|
||||
|
||||
def rmrf(path):
|
||||
shutil.rmtree(path, ignore_errors=True)
|
||||
|
||||
def feed_checksum(checksum, stream):
|
||||
b = stream.read(8192)
|
||||
while b != '':
|
||||
checksum.update(b)
|
||||
b = stream.read(8192)
|
||||
|
||||
def _find_current_origin_refspec():
|
||||
dpath = '/ostree/deploy/%s/deploy' % (os_release_data['ID'], )
|
||||
for name in os.listdir(dpath):
|
||||
if name.endswith('.origin'):
|
||||
for line in open(os.path.join(dpath, name)):
|
||||
if line.startswith('refspec='):
|
||||
return line[len('refspec='):]
|
||||
return None
|
||||
|
||||
def _initfs(targetroot):
|
||||
os.makedirs(targetroot)
|
||||
for d in ['dev', 'proc', 'run', 'sys', 'var']:
|
||||
os.mkdir(os.path.join(targetroot, d))
|
||||
|
||||
# Special ostree mount
|
||||
os.mkdir(os.path.join(targetroot, 'sysroot'))
|
||||
|
||||
# Some FHS targets; these all live in /var
|
||||
for (target, name) in [('var/opt', 'opt'),
|
||||
('var/srv', 'srv'),
|
||||
('var/mnt', 'mnt'),
|
||||
('var/roothome', 'root'),
|
||||
('var/home', 'home'),
|
||||
('run/media', 'media'),
|
||||
('sysroot/ostree', 'ostree'),
|
||||
('sysroot/tmp', 'tmp')]:
|
||||
os.symlink(target, os.path.join(targetroot, name))
|
||||
|
||||
|
||||
def _clone_current_root_to_yumroot(yumroot):
|
||||
_initfs(yumroot)
|
||||
for d in ['boot', 'usr', 'etc', 'var/lib/rpm', 'var/cache/yum']:
|
||||
destdir = os.path.join(yumroot, d)
|
||||
rmrf(destdir)
|
||||
ensuredir(os.path.dirname(destdir))
|
||||
subprocess.check_call(['cp', '--reflink=auto', '-a',
|
||||
'/' + d,
|
||||
destdir])
|
||||
|
||||
def replace_nsswitch(target_usretc):
|
||||
nsswitch_conf = os.path.join(target_usretc, 'nsswitch.conf')
|
||||
f = open(nsswitch_conf)
|
||||
newf = open(nsswitch_conf + '.tmp', 'w')
|
||||
passwd_re = re.compile(r'^passwd:\s+files(.*)$')
|
||||
group_re = re.compile(r'^group:\s+files(.*)$')
|
||||
for line in f:
|
||||
match = passwd_re.match(line)
|
||||
if match and line.find('altfiles') == -1:
|
||||
newf.write('passwd: files altfiles' + match.group(1) + '\n')
|
||||
continue
|
||||
match = group_re.match(line)
|
||||
if match and line.find('altfiles') == -1:
|
||||
newf.write('group: files altfiles' + match.group(1) + '\n')
|
||||
continue
|
||||
newf.write(line)
|
||||
f.close()
|
||||
newf.close()
|
||||
os.rename(nsswitch_conf + '.tmp', nsswitch_conf)
|
||||
|
||||
def do_kernel_prep(yumroot, logs_lookaside):
|
||||
bootdir = os.path.join(yumroot, 'boot')
|
||||
kernel_path = None
|
||||
for name in os.listdir(bootdir):
|
||||
if name.startswith('vmlinuz-'):
|
||||
kernel_path = os.path.join(bootdir, name)
|
||||
break
|
||||
elif name.startswith('initramfs-'):
|
||||
# If somehow the %post generated an initramfs, blow it
|
||||
# away - we take over that role.
|
||||
initramfs_path = os.path.join(bootdir, name)
|
||||
log("Removing RPM-generated " + initramfs_path)
|
||||
rmrf(initramfs_path)
|
||||
|
||||
if kernel_path is None:
|
||||
raise ValueError("Failed to find vmlinuz- in " + bootdir)
|
||||
|
||||
kname = os.path.basename(kernel_path)
|
||||
kver = kname[kname.find('-') + 1:]
|
||||
log("Kernel version is " + kver)
|
||||
|
||||
# OSTree will take care of this
|
||||
loaderdir = os.path.join(bootdir, 'loader')
|
||||
rmrf(loaderdir)
|
||||
|
||||
args = ['chroot', yumroot, 'depmod', kver]
|
||||
log("Running: %s" % (subprocess.list2cmdline(args), ))
|
||||
subprocess.check_call(args)
|
||||
|
||||
# Copy of code from gnome-continuous; yes, we hardcode
|
||||
# the machine id for now, because distributing pre-generated
|
||||
# initramfs images with dracut/systemd at the moment
|
||||
# effectively requires this.
|
||||
# http://lists.freedesktop.org/archives/systemd-devel/2013-July/011770.html
|
||||
log("Hardcoding machine-id")
|
||||
f = open(os.path.join(yumroot, 'etc', 'machine-id'), 'w')
|
||||
f.write('45bb3b96146aa94f299b9eb43646eb35\n')
|
||||
f.close()
|
||||
|
||||
args = ['chroot', yumroot,
|
||||
'dracut', '-v', '--tmpdir=/tmp',
|
||||
'-f', '/tmp/initramfs.img', kver];
|
||||
log("Running: %s" % (subprocess.list2cmdline(args), ))
|
||||
subprocess.check_call(args)
|
||||
|
||||
initramfs_path = os.path.join(yumroot, 'tmp', 'initramfs.img')
|
||||
if not os.path.exists(initramfs_path):
|
||||
raise ValueError("Failed to find " + initramfs_path)
|
||||
|
||||
os.rename(initramfs_path, os.path.join(bootdir, 'initramfs-' + kver + '.img'))
|
||||
varlog_dracut_path = os.path.join(yumroot, 'var', 'log', 'dracut.log')
|
||||
if os.path.exists(varlog_dracut_path):
|
||||
os.rename(varlog_dracut_path, os.path.join(logs_lookaside, 'dracut.log'))
|
||||
|
||||
def _create_rootfs_from_yumroot_content(targetroot, yumroot):
|
||||
"""Prepare a root filesystem, taking mainly the contents of /usr from yumroot"""
|
||||
|
||||
_initfs(targetroot)
|
||||
|
||||
# We take /usr from the yum content
|
||||
os.rename(os.path.join(yumroot, 'usr'), os.path.join(targetroot, 'usr'))
|
||||
# Plus the RPM database goes in usr/share/rpm
|
||||
legacyrpm_path = os.path.join(yumroot, 'var/lib/rpm')
|
||||
newrpm_path = os.path.join(targetroot, 'usr/share/rpm')
|
||||
if not os.path.isdir(newrpm_path):
|
||||
os.rename(legacyrpm_path, newrpm_path)
|
||||
|
||||
# Except /usr/local -> ../var/usrlocal
|
||||
target_usrlocal = os.path.join(targetroot, 'usr/local')
|
||||
if not os.path.islink(target_usrlocal):
|
||||
rmrf(target_usrlocal)
|
||||
os.symlink('../var/usrlocal', target_usrlocal)
|
||||
target_usretc = os.path.join(targetroot, 'usr/etc')
|
||||
rmrf(target_usretc)
|
||||
os.rename(os.path.join(yumroot, 'etc'), target_usretc)
|
||||
|
||||
# Move boot, but rename the kernel/initramfs to have a checksum
|
||||
targetboot = os.path.join(targetroot, 'boot')
|
||||
os.rename(os.path.join(yumroot, 'boot'), targetboot)
|
||||
kernel = None
|
||||
initramfs = None
|
||||
for name in os.listdir(targetboot):
|
||||
if name.startswith('vmlinuz-'):
|
||||
kernel = os.path.join(targetboot, name)
|
||||
elif name.startswith('initramfs-'):
|
||||
initramfs = os.path.join(targetboot, name)
|
||||
|
||||
assert (kernel is not None and initramfs is not None)
|
||||
|
||||
checksum = GLib.Checksum.new(GLib.ChecksumType.SHA256)
|
||||
f = open(kernel)
|
||||
feed_checksum(checksum, f)
|
||||
f.close()
|
||||
f = open(initramfs)
|
||||
feed_checksum(checksum, f)
|
||||
f.close()
|
||||
|
||||
bootcsum = checksum.get_string()
|
||||
|
||||
os.rename(kernel, kernel + '-' + bootcsum)
|
||||
os.rename(initramfs, initramfs + '-' + bootcsum)
|
||||
|
||||
# Also carry along toplevel compat links
|
||||
for name in ['lib', 'lib64', 'bin', 'sbin']:
|
||||
src = os.path.join(yumroot, name)
|
||||
if os.path.islink(src):
|
||||
os.rename(src, os.path.join(targetroot, name))
|
||||
|
||||
target_tmpfilesd = os.path.join(targetroot, 'usr/lib/tmpfiles.d')
|
||||
ensuredir(target_tmpfilesd)
|
||||
shutil.copy(os.path.join(PKGLIBDIR, 'tmpfiles-ostree-integration.conf'), target_tmpfilesd)
|
||||
|
||||
def main():
|
||||
parser = optparse.OptionParser('%prog ROOTFS REFNAME')
|
||||
parser.add_option('', "--repo",
|
||||
action='store', dest='repo_path',
|
||||
default=None,
|
||||
help="Path to OSTree repository (default=/ostree)")
|
||||
parser.add_option('-m', "--message",
|
||||
action='store', dest='message',
|
||||
default="",
|
||||
help="Commit message")
|
||||
parser.add_option('', "--gpg-sign",
|
||||
action='store',
|
||||
default=None,
|
||||
help="Sign commits using given GPG key ID")
|
||||
|
||||
global opts
|
||||
global args
|
||||
(opts, args) = parser.parse_args(sys.argv[1:])
|
||||
|
||||
rootfs_path = args[0]
|
||||
refname = args[1]
|
||||
|
||||
rootfs_tmp = rootfs_path + '.tmp'
|
||||
rmrf(rootfs_tmp)
|
||||
_create_rootfs_from_yumroot_content(rootfs_tmp, rootfs_path)
|
||||
rmrf(rootfs_path)
|
||||
os.rename(rootfs_tmp, rootfs_path)
|
||||
|
||||
if opts.repo_path is not None:
|
||||
repo = OSTree.Repo.new(Gio.File.new_for_path(opts.repo_path))
|
||||
else:
|
||||
repo = OSTree.Repo.new_default()
|
||||
|
||||
repo.open(None)
|
||||
|
||||
# To make SELinux work, we need to do the labeling right before this.
|
||||
# This really needs some sort of API, so we can apply the xattrs as
|
||||
# we're committing into the repo, rather than having to label the
|
||||
# physical FS.
|
||||
# For now, no xattrs (and hence no SELinux =( )
|
||||
log("Committing " + rootfs_path + "...")
|
||||
repo.prepare_transaction(None)
|
||||
mtree = OSTree.MutableTree.new()
|
||||
modifier = OSTree.RepoCommitModifier.new(OSTree.RepoCommitModifierFlags.SKIP_XATTRS, None, None)
|
||||
repo.write_directory_to_mtree(Gio.File.new_for_path(rootfs_path),
|
||||
mtree, modifier, None)
|
||||
[success, parent] = repo.resolve_rev(refname, True)
|
||||
[success, tree] = repo.write_mtree(mtree, None)
|
||||
[success, commit] = repo.write_commit(parent, '', opts.message, None, tree, None)
|
||||
if opts.gpg_sign is not None:
|
||||
repo.sign_commit(commit, opts.gpg_sign, None, None)
|
||||
repo.transaction_set_ref(None, refname, commit)
|
||||
repo.commit_transaction(None)
|
||||
|
||||
log("%s => %s" % (refname, commit))
|
||||
|
||||
rmrf(rootfs_path)
|
Loading…
Reference in New Issue
Block a user