12.1.4 Использование rpm2cpio для извлечения файлов из нагрузки пакетов

В стандартной ситуации сборщик пакетов имеет дело с пакетами с исходным кодом и бинарными rpm-пакетами, которые содержат файлы программ или кода в виде, пригодном для использования конечным пользователем. Порой, однако, конечный пользователь может попасть в ситуацию, когда они со сборщиком пакетов находятся по разные стороны баррикады. Доступ к файлам, содержащимся в полезной нагрузке пакета всегда можно получить, установив его в систему, но если пользователю это не нужно? Странно устанавливать пакет ради доступа к spec-файлу или одному патчу. А если нужно использовать файлы на не-Linux системе, например, в Solaris?

К счастью относительно несложно извлечь файлы из rpm-пакета с помощью специальных средств (или двух нажатий клавиши в файловом менеджере Midnight Commander - прим. переводчика).

Структурно пакет представляет собой метаданные и cpio-архив, упакованные в один файл. Если убрать бинарный заголовок, можно получить полноценный cpio-архив формата cpio System V R4, которым впоследствии манипулировать с помощью команды cpio.

RPM поставляется с утилитой rpm2cpio, основной задачей которой является конвертация файла rpm-пакета в cpio-архив, который она выдает на стандартный вывод. Базовый синтаксис использования rpm2cpio:

$ rpm2cpio fluxbox-0.1.8-2.src.rpm > fluxbox-0.1.8-2.cpio

Эта команда создает cpio-архив из пакета. Им можно манипулировать с помощью cpio или перенаправить на вход различных конвейеров:

$ rpm2cpio fluxbox-0.1.8-2.src.rpm | cpio -i -d

656 blocks

Эта команда получает содержимое нагрузки пакета в виде файлов.

rpm2cpio входит в состав RPM и поэтому доступна во всех rpm-based Linux дистрибутивах. Однако, стандартная реализация утилиты написана на C, поэтому ее использование, например, в Solaris, подразумевает первоначально сборку бинарного модуля. Для кросс-платформенного применения существует две реализации rpm2cpio на скриптовых языках - Perl и bash. Bash скрипт будет работать не только в большинстве Unix-системах, но и в Windows под cygwin. Ниже приведен скрипт, который можно сохранить под именем rpm2cpio.sh, сделать исполнимым и поместить в стандартный каталог программ, например, /usr/bin, после чего можно манипулировать архивами, полученными из rpm-пакетов обычным образом:

#!/bin/sh

pkg=$1

if [ "$pkg" = "" -o ! -e "$pkg" ]; then

echo "no package supplied" 1>&2

exit 1

fi

leadsize=96

o=`expr $leadsize + 8`

set `od -j $o -N 8 -t u1 $pkg`

il=`expr 256 \* \( 256 \* \( 256 \* $2 + $3 \) + $4 \) + $5`

dl=`expr 256 \* \( 256 \* \( 256 \* $6 + $7 \) + $8 \) + $9`

# echo "sig il: $il dl: $dl"

sigsize=`expr 8 + 16 \* $il + $dl`

o=`expr $o + $sigsize + \( 8 - \( $sigsize \% 8 \) \) \% 8 + 8`

set `od -j $o -N 8 -t u1 $pkg`

il=`expr 256 \* \( 256 \* \( 256 \* $2 + $3 \) + $4 \) + $5`

dl=`expr 256 \* \( 256 \* \( 256 \* $6 + $7 \) + $8 \) + $9`

# echo "hdr il: $il dl: $dl"

hdrsize=`expr 8 + 16 \* $il + $dl`

o=`expr $o + $hdrsize`

dd if=$pkg ibs=$o skip=1 2>/dev/null | gunzip

Поскольку скрипт выдает результат на стандартный вывод, потребуется перенаправление, также как в случае C-реализации:

$ rpm2cpio.sh fluxbox-0.1.8-2.src.rpm | cpio -i -d

656 blocks

В данном случае мы перенаправили вывод на cpio, распаковывая архив напрямую. Можно перенаправить вывод и в файл, если это требуется.

Perl-реализация rpm2cpio написана Roger Espel Llima. Этот скрипт должен работать во всех системах, имеющих современный интерпретатор Perl. Сохраните скрипт в файле rpm2cpio.pl, сделайте его исполнимым и поместите в каталог программ.

#!/usr/bin/perl

# Copyright (C) 1997,1998,1999, Roger Espel Llima

#

# Permission is hereby granted, free of charge, to any person obtaining a copy

# of this software and any associated documentation files (the "Software"), to

# deal in the Software without restriction, including without limitation the

# rights to use, copy, modify, merge, publish, distribute, sublicense,

# and/or sell copies of the Software, and to permit persons to whom the

# Software is furnished to do so, subject to the following conditions:

#

# The above copyright notice and this permission notice shall be included in

# all copies or substantial portions of the Software.

#

# THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR

# IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,

# FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE

# SOFTWARE'S COPYRIGHT HOLDER(S) BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER

# LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,

# OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN

# THE SOFTWARE

# (whew, that's done!)

# why does the world need another rpm2cpio? because the existing one

# won't build unless you have half a ton of things that aren't really

# required for it, since it uses the same library used to extract RPM's.

# in particular, it won't build on the HPsUX box i'm on.

# sw 2002-Mar-6 Don't slurp the whole file

# add a path if desired

$gzip = "gzip";

sub printhelp {

print <
rpm2cpio, perl version by orabidoo +sw

dumps the contents to stdout as a cpio archive

use: rpm2cpio [file.rpm] > file.cpio

Here's how to use cpio:

list of contents: cpio -t -i < /file/name

extract files: cpio -d -i < /file/name

HERE

exit 0;

}

if ($#ARGV == -1) {

printhelp if -t STDIN;

$f = "STDIN";

} elsif ($#ARGV == 0) {

open(F, "< $ARGV[0]") or die "Can't read file $ARGV[0]\n";

$f = 'F';

} else {

printhelp;

}

printhelp if -t STDOUT;

# gobble the file up

##undef $/;

##$|=1;

##$rpm = <$f>;

##close ($f);

read $f,$rpm,96;

($magic, $major, $minor, $crap) = unpack("NCC C90", $rpm);

die "Not an RPM\n" if $magic != 0xedabeedb;

die "Not a version 3 or 4 RPM\n" if $major != 3 && $major != 4;

##$rpm = substr($rpm, 96);

while (!eof($f)) {

$pos = tell($f);

read $f,$rpm,16;

$smagic = unpack("n", $rpm);

last if $smagic eq 0x1f8b;

# Turns out that every header except the start of the gzip one is

# padded to an 8 bytes boundary.

if ($pos & 0x7) {

$pos += 7;

$pos &= ~0x7;# Round to 8 byte boundary

seek $f, $pos, 0;

read $f,$rpm,16;

}

($magic, $crap, $sections, $bytes) = unpack("N4", $rpm);

die "Error: header not recognized\n" if $magic != 0x8eade801;

$pos += 16;# for header

$pos += 16 * $sections;

$pos += $bytes;

seek $f, $pos, 0;

}

if (eof($f)) {

die "bogus RPM\n";

}

open(ZCAT, "|gzip -cd") || die "can't pipe to gzip\n";

print STDERR "CPIO archive found!\n";

print ZCAT $rpm;

while (read($f, ($_=''), 16384) > 0) {

print ZCAT;

}

close ZCAT;

Применение rpm2spio.pl:

$ rpm2cpio.pl fluxbox-0.1.8-2.src.rpm | cpio -i -d

CPIO archive found!

656 blocks

В зависимости от используемой системы, какая-то из трех реализаций будет работоспособна (или все).

Далее - Раздел 13. Пакетостроение - генеральная линия партии или колхоз "Светлый путь"
Назад - Отладка spec-файла с помощью rpmlint
Содержание