В стандартной ситуации сборщик пакетов имеет дело с пакетами с исходным кодом и бинарными 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 
 Содержание