You can not select more than 25 topics Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
tdeadmin/kpackage/debInterface.cpp

597 lines
14 KiB

/*
** Copyright (C) 1999,2000 Toivo Pedaste <toivo@ucs.uwa.edu.au>
**
*/
/*
** This program 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 program 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 in a file called COPYING; if not, write to
** the Free Software Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston,
** MA 02110-1301, USA.
*/
/*
** Bug reports and questions can be sent to kde-devel@kde.org
*/
#include "../config.h"
#include <unistd.h>
#include <stdlib.h> // for getenv
#include <sys/types.h>
#include <sys/stat.h>
#include <fcntl.h> // for O_RDONLY
#include <setjmp.h>
#include <iostream>
#include <kurl.h>
#include <kglobal.h>
#include <kiconloader.h>
#include <kdebug.h>
#include "packageInfo.h"
#include "debInterface.h"
#include "updateLoc.h"
#include "kpackage.h"
#include "managementWidget.h"
#include "utils.h"
#include "options.h"
#include "cache.h"
#include <klocale.h>
extern KApplication *app;
//////////////////////////////////////////////////////////////////////////////
//////////////////////////////////////////////////////////////////////////////
DEB::DEB():pkgInterface()
{
head = "DEB";
icon = "deb";
pict = UserIcon(icon);
bad_pict = UserIcon("dbad");
updated_pict = UserIcon("dupdated");
new_pict = UserIcon("dnew");
packagePattern = "*.deb";
typeID = "/deb";
locatedialog = 0;
}
//////////////////////////////////////////////////////////////////////////////
//////////////////////////////////////////////////////////////////////////////
DEB::~DEB()
{
}
//////////////////////////////////////////////////////////////////////////////
// check if debian file
bool DEB::isType(char *buf, const TQString &)
{
if (hasProgram) {
if (!strcmp(buf,"!<arch>\n")) {
return true;
} else if (!strncmp(buf,"0.9",3)) {
return true;
} else
return false;
} else {
return false;
}
}
void DEB::distPackages(TQPtrList<packageInfo> *, cacheObj *)
{
}
void DEB::listUnIPackages(TQPtrList<packageInfo> *pki, LcacheObj *pCache)
{
TQString s;
cacheObj *cp;
for (cp = pCache->first(); cp != 0; cp = pCache->next()) {
kdDebug() << cp->base << ":: " << cp->option << "\n";
if (!cp->option.isEmpty()) {
distPackages(pki, cp);
} else if (!cp->base.isEmpty()) {
s = getPackList(cp);
if (!s.isEmpty()) {
listPackList(pki,s,cp);
}
} else {
s = getDir(cp);
if (!s.isEmpty()) {
listDir(pki,s,cp->location,cp->subdirs);
}
}
}
}
bool DEB::parseName(const TQString &name, TQString *n, TQString *v)
{
int d1, d2, s1;
s1 = name.tqfindRev('.');
if (s1 > 0) {
d2 = name.tqfindRev('-',s1-1);
if (d2 > 0) {
d1 = name.tqfindRev('_',d2-1);
if (d1 < 0)
d1 = d2;
*n = name.left(d1);
*v = name.mid(d1+1,s1-d1-1);
return TRUE;
}
}
return FALSE;
}
void DEB::listInstalledPackages(TQPtrList<packageInfo> *pki)
{
listPackList(pki,STATUS,0);
}
void DEB::listPackList(TQPtrList<packageInfo> *pki, const TQString &fname, cacheObj *cp)
{
bool local = FALSE;
packageInfo *p;
TQStringList list;
TQString sline( i18n("Querying DEB package list: ")+fname );
if (cp) {
KURL u(cp->base);
local = u.isLocalFile();
}
kpackage->settqStatus(sline);
kpackage->setPercent(0);
TQFile file(STATUS);
TQString s;
bool fileOpened= file.open(IO_ReadOnly);
if (fileOpened) {
TQTextStream stream( &file );
s = "";
while ( !s.isNull() ) {
s = stream.readLine();
if ( !s.isEmpty() ) {
list << s;
} else {
p = collectInfo(list);
if (p) {
if (!p->pkgInsert(pki, typeID, cp == 0)) {
delete p;
} else if (cp) {
p->info.insert("base", cp->base);
}
}
list.clear();
}
}
file.close();
}
kpackage->setPercent(100);
}
//////////////////////////////////////////////////////////////////////////////
//////////////////////////////////////////////////////////////////////////////
packageInfo *DEB::getPackageInfo(char mode, const TQString &name, const TQString &)
{
if (mode == 'i') {
if (hostName.isEmpty()) {
return getIPackageInfo(name);
} else {
return getIRPackageInfo(name);
}
} else
return getUPackageInfo(name);
}
packageInfo *DEB::getIPackageInfo( const TQString &name)
{
// query an installed package!
packageInfo *pki = 0;
TQString search;
TQStringList list;
search = "Package: "+ name;
TQFile f(STATUS);
if ( f.open(IO_ReadOnly) ) {
TQTextStream t( &f );
TQString s;
while ( !t.eof() ) {
s = t.readLine();
if ( s == search) {
list << s;
break;
}
}
while ( !t.eof() ) {
s = t.readLine();
if (s.length()) {
list << s;
} else {
pki = collectInfo(list);
break;
}
}
}
f.close();
return pki;
}
packageInfo *DEB::getIRPackageInfo( const TQString &name)
{
// query an remote installed package
packageInfo *pki = 0;
TQString s = "dpkg --status ";
s += name;
TQStringList list = kpty->run(s);
for ( TQStringList::Iterator it = list.begin(); it != list.end(); ++it ) {
// kdDebug() << "U=" << *it << "\n";
if ((*it).tqfind("Package:") >= 0) {
kdDebug() << "found\n";
while (it != list.begin()) {
list.remove(list.begin());
}
break;
}
}
if (list.count() > 1) {
pki = DEB::collectInfo(list);
if (pki) {
pki->updated = TRUE;
if (pki->getFilename().isEmpty())
pki->setFilename(name);
}
}
return pki;
}
packageInfo *DEB::getUPackageInfo( const TQString &name)
{
// query an uninstalled package
packageInfo *pki = 0;
TQString s = "dpkg --info ";
s += KProcess::quote(name);
TQStringList list = kpty->run(s);
for ( TQStringList::Iterator it = list.begin(); it != list.end(); ++it ) {
// kdDebug() << "U=" << *it << "\n";
if ((*it).tqfind("Package:") >= 0) {
// kdDebug() << "found\n";
while (it != list.begin()) {
list.remove(list.begin());
}
break;
}
}
if (list.count() > 1) {
pki = DEB::collectInfo(list, kpinterface[0]); // To be fixed up later, assumes order of kpinterface
if (pki)
pki->updated = TRUE;
}
return pki;
}
//////////////////////////////////////////////////////////////////////////////
//////////////////////////////////////////////////////////////////////////////
packageInfo *DEB::collectInfo(TQStringList &ln, pkgInterface *pkgInt)
{
if (!pkgInt) {
pkgInt = this;
}
TQMap<TQString, TQString> a;
TQString key, val;
bool bad_install = FALSE;
bool available = FALSE;
bool haveName = FALSE;
for ( TQStringList::Iterator it = ln.begin(); it != ln.end(); ++it ) {
loop:
int col = (*it).tqfind(':');
key = ((*it).left(col)).lower();
if (key[0] == ' ') {
key.remove(0,1);
}
val = (*it).mid(col+2);
// val.truncate(val.length() - 1);
if (key == "conffiles") {
while (++it != ln.end()) {
if ((*it)[0] == ' ') {
} else {
goto loop;
}
}
} else if (key == "description") {
a.insert("summary", val);
TQString desc;
while (++it != ln.end()) {
if ((*it)[0] == ' ') {
desc += *it;
} else {
a.insert("description", desc);
goto loop;
}
}
a.insert("description", desc);
break;
} else if (key == "package") {
a.insert("name", val);
haveName = TRUE;
} else if (key == "md5sum") {
available = TRUE;
bad_install = FALSE;
} else if (key == "section") {
a.insert("group", val);
} else if (key == "status") {
if ((val.tqfind("not-installed") >= 0) || (val.tqfind("config-files") >= 0)) {
return 0;
}
if (val != "install ok installed" &&
val != "deinstall ok installed" &&
val != "deinstall ok config-files" &&
val != "purge ok installed") {
bad_install = TRUE;
}
a.insert("status", val);
} else if (key == "version") {
a.insert("version", val);
} else if (key == "size") {
a.insert("file-size", val);
} else if (key == "installed-size") {
a.insert("size", val + "000");
} else {
a.insert(key, val);
}
// kdDebug() << "C=" << key << "," << val <<"\n";
}
if (haveName) {
packageInfo *i = new packageInfo(a,pkgInt);
if (bad_install) {
i->packageState = packageInfo::BAD_INSTALL;
} else if (available) {
i->packageState = packageInfo::AVAILABLE;
} else {
i->packageState = packageInfo::INSTALLED;
}
i->fixup();
return i;
} else {
return 0;
}
}
//////////////////////////////////////////////////////////////////////////////
TQStringList DEB::getChangeLog(packageInfo *p) {
TQString fn( p->getFilename());
if(!fn.isEmpty())
return 0;
else
return getIChangeLog(p);
}
TQStringList DEB::getIChangeLog(packageInfo *p)
{
TQString from;
TQStringList ret;
TQString name = p->getProperty("name");
from = "zcat /usr/share/doc/";
from += name;
from += "/changelog.Debian.gz";
ret = kpty->run(from);
if (!kpty->Result)
return ret;
else {
from = "zcat /usr/share/doc/";
from += name;
from += "/changelog.gz";
ret = kpty->run(from);
if (!kpty->Result)
return ret;
else
return 0;
}
}
bool DEB::filesTab(packageInfo *p) {
if (p->packageState == packageInfo::INSTALLED) {
return true;
} else if (p->isFileLocal()) {
return true;
}
return false;
}
bool DEB::changeTab(packageInfo *p) {
if (p->packageState == packageInfo::INSTALLED) {
return true;
}
return false;
}
//////////////////////////////////////////////////////////////////////////////
TQStringList DEB::getFileList(packageInfo *p)
{
TQString fn( p->getFilename());
if(!fn.isEmpty())
return getUFileList(fn);
else
return getIFileList(p);
}
// query an installed package
TQStringList DEB::getIFileList(packageInfo *p)
{
FILE *file;
TQString name = p->getProperty("name");
TQStringList filelist;
TQString vb( INFODIR + name + ".list");
file= fopen(vb.ascii(),"r");
if (file) {
char linebuf[1024];
while (fgets(linebuf,sizeof(linebuf),file)) {
linebuf[strlen(linebuf) - 1] = 0; // remove new line
filelist.append(linebuf);
}
fclose(file);
}
return filelist;
}
// query an uninstalled package
TQStringList DEB::getUFileList(const TQString &fn)
{
TQString s = "dpkg --contents ";
s += "'";
s += fn;
s += "'";
TQStringList filelist = kpty->run(s);
int pt = -1;
for ( TQStringList::Iterator it = filelist.begin();
it != filelist.end(); ++it ) {
// kdDebug() << "F=" << *it << "\n";
if (pt < 0) {
pt = (*it).tqfindRev(' ');
}
(*it) = (*it).mid(pt + 1);
}
return filelist;
}
//////////////////////////////////////////////////////////////////////////////
//////////////////////////////////////////////////////////////////////////////
TQStringList DEB::FindFile(const TQString &name, bool)
{
TQString s = "dpkg -S ";
s += name;
TQStringList filelist = kpty->run(s);
for ( TQStringList::Iterator it = filelist.begin(); it != filelist.end(); ++it ) {
int p = (*it).tqfind(": ");
if( p !=-1 )
(*it).tqreplace(p, 2, "\t");
}
if (filelist.count() == 1) {
TQStringList::Iterator it = filelist.begin();
if ((*it).tqfind("not found") >= 0) {
filelist.remove(it);
}
}
return filelist;
}
//////////////////////////////////////////////////////////////////////////////
//////////////////////////////////////////////////////////////////////////////
void DEB::setLocation()
{
locatedialog->restore();
}
//////////////////////////////////////////////////////////////////////////////
//////////////////////////////////////////////////////////////////////////////
void DEB::setAvail(LcacheObj *slist)
{
if (packageLoc)
delete packageLoc;
packageLoc = slist;
}
//////////////////////////////////////////////////////////////////////////////
//////////////////////////////////////////////////////////////////////////////
TQString DEB::uninstall(int uninstallFlags, TQPtrList<packageInfo> *p,
bool &test)
{
TQString packs = "";
packageInfo *i;
for (i = p->first(); i!= 0; i = p->next()) {
packs += i->getProperty("name");
packs += " ";
}
return doUninstall( uninstallFlags, packs, test);
}
//////////////////////////////////////////////////////////////////////////////
TQString DEB::install(int installFlags, TQPtrList<packageInfo> *p,
bool &test)
{
TQString packs = "";
packageInfo *i;
for (i = p->first(); i!= 0; i = p->next()) {
TQString fname = i->fetchFilename();
if (!fname.isEmpty()) {
packs += KProcess::quote(fname);
packs += " ";
}
}
return doInstall(installFlags, packs, test);
}
//////////////////////////////////////////////////////////////////////////////
// Call the script to install packages setting parameters
// to dpkg dependent on flags, returning whether everyting worked
//////////////////////////////////////////////////////////////////////////////
//////////////////////////////////////////////////////////////////////////////
TQString DEB::doInstall(int installFlags, const TQString &packs, bool &test)
{
TQString s = "dpkg -i ";
s += setOptions(installFlags, paramsInst);
s += packs;
kdDebug() << "iCMD=" << s << "\n";
if (installFlags>>4 & 1)
test = 1;
return s;
}
#include "debInterface.moc"