blob: 1dd124e599587f9d7b9144fdf85f63724e034397 [file] [log] [blame]
/*
* dpkg-split - splitting and joining of multipart *.deb archives
* join.c - joining
*
* Copyright © 1995 Ian Jackson <ian@chiark.greenend.org.uk>
*
* This is free software; you can redistribute it and/or modify
* it under the terms of the GNU General Public License as published by
* the Free Software Foundation; either version 2 of the License, or
* (at your option) any later version.
*
* This 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 General Public License for more details.
*
* You should have received a copy of the GNU General Public License
* along with this program. If not, see <http://www.gnu.org/licenses/>.
*/
#include <config.h>
#include <compat.h>
#include <assert.h>
#include <limits.h>
#include <string.h>
#include <fcntl.h>
#include <unistd.h>
#include <stdlib.h>
#include <stdio.h>
#include <dpkg/i18n.h>
#include <dpkg/dpkg.h>
#include <dpkg/dpkg-db.h>
#include <dpkg/buffer.h>
#include <dpkg/options.h>
#include "dpkg-split.h"
void reassemble(struct partinfo **partlist, const char *outputfile) {
int fd_out, fd_in;
unsigned int i;
printf(P_("Putting package %s together from %d part: ",
"Putting package %s together from %d parts: ",
partlist[0]->maxpartn),
partlist[0]->package,partlist[0]->maxpartn);
fd_out = creat(outputfile, 0644);
if (fd_out < 0)
ohshite(_("unable to open output file `%.250s'"), outputfile);
for (i=0; i<partlist[0]->maxpartn; i++) {
struct partinfo *pi = partlist[i];
fd_in = open(pi->filename, O_RDONLY);
if (fd_in < 0)
ohshite(_("unable to (re)open input part file `%.250s'"), pi->filename);
fd_skip(fd_in, pi->headerlen, _("skipping split package header"));
fd_fd_copy(fd_in, fd_out, pi->thispartlen, _("split package part"));
close(fd_in);
printf("%d ",i+1);
}
if (fsync(fd_out))
ohshite(_("unable to sync file '%s'"), outputfile);
if (close(fd_out))
ohshite(_("unable to close file '%s'"), outputfile);
printf(_("done\n"));
}
void addtopartlist(struct partinfo **partlist,
struct partinfo *pi, struct partinfo *refi) {
int i;
if (strcmp(pi->package,refi->package) ||
strcmp(pi->version,refi->version) ||
strcmp(pi->md5sum,refi->md5sum) ||
pi->orglength != refi->orglength ||
pi->maxpartn != refi->maxpartn ||
pi->maxpartlen != refi->maxpartlen) {
print_info(pi);
print_info(refi);
ohshit(_("files `%.250s' and `%.250s' are not parts of the same file"),
pi->filename,refi->filename);
}
i= pi->thispartn-1;
if (partlist[i])
ohshit(_("there are several versions of part %d - at least `%.250s' and `%.250s'"),
pi->thispartn, pi->filename, partlist[i]->filename);
partlist[i]= pi;
}
int
do_join(const char *const *argv)
{
const char *thisarg;
struct partqueue *pq;
struct partinfo *refi, **partlist;
unsigned int i;
assert(!queue);
if (!*argv)
badusage(_("--%s requires one or more part file arguments"),
cipaction->olong);
while ((thisarg= *argv++)) {
pq= nfmalloc(sizeof(struct partqueue));
mustgetpartinfo(thisarg,&pq->info);
pq->nextinqueue= queue;
queue= pq;
}
refi= NULL;
for (pq= queue; pq; pq= pq->nextinqueue)
if (!refi || pq->info.thispartn < refi->thispartn) refi= &pq->info;
assert(refi);
partlist= nfmalloc(sizeof(struct partinfo*)*refi->maxpartn);
for (i = 0; i < refi->maxpartn; i++)
partlist[i] = NULL;
for (pq= queue; pq; pq= pq->nextinqueue) {
struct partinfo *pi = &pq->info;
addtopartlist(partlist,pi,refi);
}
for (i=0; i<refi->maxpartn; i++) {
if (!partlist[i]) ohshit(_("part %d is missing"),i+1);
}
if (!opt_outputfile) {
char *p;
p= nfmalloc(strlen(refi->package)+1+strlen(refi->version)+sizeof(DEBEXT));
strcpy(p,refi->package);
strcat(p, "_");
strcat(p,refi->version);
strcat(p, "_");
strcat(p, refi->arch ? refi->arch : "unknown");
strcat(p,DEBEXT);
opt_outputfile = p;
}
reassemble(partlist, opt_outputfile);
return 0;
}