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.
tdebase/kioslave/nfs/kio_nfs.cpp

1616 lines
48 KiB

/* This file is part of the KDE project
Copyright (C) 2000 Alexander Neundorf <neundorf@kde.org>
This library is free software; you can redistribute it and/or
modify it under the terms of the GNU Library 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
Library General Public License for more details.
You should have received a copy of the GNU Library General Public License
along with this library; see the file COPYING.LIB. If not, write to
the Free Software Foundation, Inc., 51 Franklin Street, Fifth Floor,
Boston, MA 02110-1301, USA.
*/
#ifdef HAVE_CONFIG_H
#include <config.h>
#endif
#include <sys/types.h>
#include <sys/stat.h>
#include <sys/utsname.h>
#include <arpa/inet.h>
// This is needed on Solaris so that rpc.h defines clnttcp_create etc.
#ifndef PORTMAP
#define PORTMAP
#endif
#include <rpc/rpc.h> // for rpc calls
#include <errno.h>
#include <grp.h>
#include <memory.h>
#include <netdb.h>
#include <pwd.h>
#include <stdlib.h>
#include <strings.h>
#include <stdio.h>
#include <time.h>
#include <unistd.h>
#include <tqfile.h>
#include <tqdir.h>
#include <kdebug.h>
#include <kinstance.h>
#include <klocale.h>
#include <kio/global.h>
#include <iostream>
#include "nfs_prot.h"
#define fhandle _fhandle
#include "mount.h"
#include "kio_nfs.h"
#define MAXHOSTLEN 256
//#define MAXFHAGE 60*15 //15 minutes maximum age for file handles
//this ioslave is for NFS version 2
#define NFSPROG ((u_long)100003)
#define NFSVERS ((u_long)2)
using namespace KIO;
using namespace std;
//this is taken from kdelibs/tdecore/fakes.cpp
//#if !defined(HAVE_GETDOMAINNAME)
int x_getdomainname(char *name, size_t len)
{
struct utsname uts;
struct hostent *hent;
int rv = -1;
if (name == 0L)
errno = EINVAL;
else
{
name[0] = '\0';
if (uname(&uts) >= 0)
{
if ((hent = gethostbyname(uts.nodename)) != 0L)
{
char *p = strchr(hent->h_name, '.');
if (p != 0L)
{
++p;
if (strlen(p) > len-1)
errno = EINVAL;
else
{
strcpy(name, p);
rv = 0;
}
}
}
}
}
return rv;
}
//#endif
extern "C" { int KDE_EXPORT kdemain(int argc, char **argv); }
int kdemain( int argc, char **argv )
{
KInstance instance( "kio_nfs" );
if (argc != 4)
{
fprintf(stderr, "Usage: kio_nfs protocol domain-socket1 domain-socket2\n");
exit(-1);
}
kdDebug(7121) << "NFS: kdemain: starting" << endl;
NFSProtocol slave(argv[2], argv[3]);
slave.dispatchLoop();
return 0;
}
static bool isRoot(const TQString& path)
{
return (path.isEmpty() || (path=="/"));
}
static bool isAbsoluteLink(const TQString& path)
{
//hmm, don't know
if (path.isEmpty()) return TRUE;
if (path[0]=='/') return TRUE;
return FALSE;
}
static void createVirtualDirEntry(UDSEntry & entry)
{
UDSAtom atom;
atom.m_uds = KIO::UDS_FILE_TYPE;
atom.m_long = S_IFDIR;
entry.append( atom );
atom.m_uds = KIO::UDS_ACCESS;
atom.m_long = S_IRUSR | S_IXUSR | S_IRGRP | S_IXGRP | S_IROTH | S_IXOTH;
entry.append( atom );
atom.m_uds = KIO::UDS_USER;
atom.m_str = "root";
entry.append( atom );
atom.m_uds = KIO::UDS_GROUP;
atom.m_str = "root";
entry.append( atom );
//a dummy size
atom.m_uds = KIO::UDS_SIZE;
atom.m_long = 1024;
entry.append( atom );
}
static void stripTrailingSlash(TQString& path)
{
//if (path=="/") return;
if (path=="/") path="";
else if (path[path.length()-1]=='/') path.truncate(path.length()-1);
}
static void getLastPart(const TQString& path, TQString& lastPart, TQString& rest)
{
int slashPos=path.findRev("/");
lastPart=path.mid(slashPos+1);
rest=path.left(slashPos+1);
}
static TQString removeFirstPart(const TQString& path)
{
TQString result("");
if (path.isEmpty()) return result;
result=path.mid(1);
int slashPos=result.find("/");
return result.mid(slashPos+1);
}
NFSFileHandle::NFSFileHandle()
:m_isInvalid(FALSE)
{
memset(m_handle,'\0',NFS_FHSIZE+1);
// m_detectTime=time(0);
}
NFSFileHandle::NFSFileHandle(const NFSFileHandle & handle)
:m_isInvalid(FALSE)
{
m_handle[NFS_FHSIZE]='\0';
memcpy(m_handle,handle.m_handle,NFS_FHSIZE);
m_isInvalid=handle.m_isInvalid;
// m_detectTime=handle.m_detectTime;
}
NFSFileHandle::~NFSFileHandle()
{}
NFSFileHandle& NFSFileHandle::operator= (const NFSFileHandle& src)
{
memcpy(m_handle,src.m_handle,NFS_FHSIZE);
m_isInvalid=src.m_isInvalid;
// m_detectTime=src.m_detectTime;
return *this;
}
NFSFileHandle& NFSFileHandle::operator= (const char* src)
{
if (src==0)
{
m_isInvalid=TRUE;
return *this;
};
memcpy(m_handle,src,NFS_FHSIZE);
m_isInvalid=FALSE;
// m_detectTime=time(0);
return *this;
}
/*time_t NFSFileHandle::age() const
{
return (time(0)-m_detectTime);
}*/
NFSProtocol::NFSProtocol (const TQCString &pool, const TQCString &app )
:SlaveBase( "nfs", pool, app )
,m_client(0)
,m_sock(-1)
,m_lastCheck(time(0))
{
kdDebug(7121)<<"NFS::NFS: -"<<pool<<"-"<<endl;
}
NFSProtocol::~NFSProtocol()
{
closeConnection();
}
/*This one is currently unused, so it could be removed.
The intention was to keep handles around, and from time to time
remove handles which are too old. Alex
*/
/*void NFSProtocol::checkForOldFHs()
{
kdDebug(7121)<<"checking for fhs older than "<<MAXFHAGE<<endl;
kdDebug(7121)<<"current items: "<<m_handleCache.count()<<endl;
NFSFileHandleMap::Iterator it=m_handleCache.begin();
NFSFileHandleMap::Iterator lastIt=it;
while (it!=m_handleCache.end())
{
kdDebug(7121)<<it.data().age()<<flush;
if (it.data().age()>MAXFHAGE)
{
kdDebug(7121)<<"removing"<<endl;
m_handleCache.remove(it);
if (it==lastIt)
{
it=m_handleCache.begin();
lastIt=it;
}
else
it=lastIt;
}
lastIt=it;
it++;
};
kdDebug(7121)<<"left items: "<<m_handleCache.count()<<endl;
m_lastCheck=time(0);
}*/
void NFSProtocol::closeConnection()
{
close(m_sock);
m_sock=-1;
if (m_client==0) return;
CLNT_DESTROY(m_client);
m_client=0;
}
bool NFSProtocol::isExportedDir(const TQString& path)
{
return (m_exportedDirs.find(path.mid(1))!=m_exportedDirs.end());
}
/* This one works recursive.
It tries to get the file handle out of the file handle cache.
If this doesn't succeed, it needs to do a nfs rpc call
in order to obtain one.
*/
NFSFileHandle NFSProtocol::getFileHandle(TQString path)
{
if (m_client==0) openConnection();
//I'm not sure if this is useful
//if ((time(0)-m_lastCheck)>MAXFHAGE) checkForOldFHs();
stripTrailingSlash(path);
kdDebug(7121)<<"getting FH for -"<<path<<"-"<<endl;
//now the path looks like "/root/some/dir" or "" if it was "/"
NFSFileHandle parentFH;
//we didn't find it
if (path.isEmpty())
{
kdDebug(7121)<<"path is empty, invalidating the FH"<<endl;
parentFH.setInvalid();
return parentFH;
}
//check wether we have this filehandle cached
//the filehandles of the exported root dirs are always in the cache
if (m_handleCache.find(path)!=m_handleCache.end())
{
kdDebug(7121)<<"path is in the cache, returning the FH -"<<m_handleCache[path]<<"-"<<endl;
return m_handleCache[path];
}
TQString rest, lastPart;
getLastPart(path,lastPart,rest);
kdDebug(7121)<<"splitting path into rest -"<<rest<<"- and lastPart -"<<lastPart<<"-"<<endl;
parentFH=getFileHandle(rest);
//f*ck, it's invalid
if (parentFH.isInvalid())
{
kdDebug(7121)<<"the parent FH is invalid"<<endl;
return parentFH;
}
// do the rpc call
diropargs dirargs;
diropres dirres;
memcpy(dirargs.dir.data,(const char*)parentFH,NFS_FHSIZE);
TQCString tmpStr=TQFile::encodeName(lastPart);
dirargs.name=tmpStr.data();
//cerr<<"calling rpc: FH: -"<<parentFH<<"- with name -"<<dirargs.name<<"-"<<endl;
int clnt_stat = clnt_call(m_client, NFSPROC_LOOKUP,
(xdrproc_t) xdr_diropargs, (char*)&dirargs,
(xdrproc_t) xdr_diropres, (char*)&dirres,total_timeout);
if ((clnt_stat!=RPC_SUCCESS) || (dirres.status!=NFS_OK))
{
//we failed
kdDebug(7121)<<"lookup of filehandle failed"<<endl;
parentFH.setInvalid();
return parentFH;
}
//everything went fine up to now :-)
parentFH=dirres.diropres_u.diropres.file.data;
//kdDebug(7121)<<"filesize: "<<dirres.diropres_u.diropres.attributes.size<<endl;
m_handleCache.insert(path,parentFH);
kdDebug(7121)<<"return FH -"<<parentFH<<"-"<<endl;
return parentFH;
}
/* Open connection connects to the mount daemon on the server side.
In order to do this it needs authentication and calls auth_unix_create().
Then it asks the mount daemon for the exported shares. Then it tries
to mount all these shares. If this succeeded for at least one of them,
a client for the nfs daemon is created.
*/
void NFSProtocol::openConnection()
{
kdDebug(7121)<<"NFS::openConnection for -" << m_currentHost.latin1() << "-" << endl;
if (m_currentHost.isEmpty())
{
error(ERR_UNKNOWN_HOST,"");
return;
}
struct sockaddr_in server_addr;
if (m_currentHost[0] >= '0' && m_currentHost[0] <= '9')
{
server_addr.sin_family = AF_INET;
server_addr.sin_addr.s_addr = inet_addr(m_currentHost.latin1());
}
else
{
struct hostent *hp=gethostbyname(m_currentHost.latin1());
if (hp==0)
{
error( ERR_UNKNOWN_HOST, m_currentHost.latin1() );
return;
}
server_addr.sin_family = AF_INET;
memcpy(&server_addr.sin_addr, hp->h_addr, hp->h_length);
}
// create mount deamon client
closeConnection();
server_addr.sin_port = 0;
m_sock = RPC_ANYSOCK;
m_client=clnttcp_create(&server_addr,MOUNTPROG, MOUNTVERS, &m_sock, 0, 0);
if (m_client==0)
{
server_addr.sin_port = 0;
m_sock = RPC_ANYSOCK;
pertry_timeout.tv_sec = 3;
pertry_timeout.tv_usec = 0;
m_client = clntudp_create(&server_addr,MOUNTPROG, MOUNTVERS, pertry_timeout, &m_sock);
if (m_client==0)
{
clnt_pcreateerror(const_cast<char *>("mount clntudp_create"));
error(ERR_COULD_NOT_CONNECT, m_currentHost.latin1());
return;
}
}
TQCString hostName("localhost");
char nameBuffer[1024];
nameBuffer[0] = '\0';
if (gethostname(nameBuffer, 1024)==0)
{
nameBuffer[sizeof(nameBuffer)-1] = '\0';
hostName=nameBuffer;
// I have the same problem here as Stefan Westerfeld, that's why I use
// the getdomainname() from fakes.cpp (renamed to x_getdomainname()), this one works
// taken from kdelibs/arts/mcopy/mcoputils.cc
nameBuffer[0] = '\0';
if (x_getdomainname(nameBuffer, 1024)==0)
{
nameBuffer[sizeof(nameBuffer)-1] = '\0';
/*
* I don't know why, but on my linux machine, the domainname
* always ends up being (none), which is certainly no valid
* domainname
*/
if(strcmp(nameBuffer,"(none)") != 0) {
hostName += ".";
hostName += nameBuffer;
}
}
}
kdDebug(7121) << "hostname is -" << hostName << "-" << endl;
m_client->cl_auth = authunix_create(hostName.data(), geteuid(), getegid(), 0, 0);
total_timeout.tv_sec = 20;
total_timeout.tv_usec = 0;
exports exportlist;
//now do the stuff
memset(&exportlist, '\0', sizeof(exportlist));
int clnt_stat = clnt_call(m_client, MOUNTPROC_EXPORT,(xdrproc_t) xdr_void, NULL,
(xdrproc_t) xdr_exports, (char*)&exportlist,total_timeout);
if (!checkForError(clnt_stat, 0, m_currentHost.latin1())) return;
fhstatus fhStatus;
bool atLeastOnceSucceeded(FALSE);
for(; exportlist!=0;exportlist = exportlist->ex_next) {
kdDebug(7121) << "found export: " << exportlist->ex_dir << endl;
memset(&fhStatus, 0, sizeof(fhStatus));
clnt_stat = clnt_call(m_client, MOUNTPROC_MNT,(xdrproc_t) xdr_dirpath, (char*)(&(exportlist->ex_dir)),
(xdrproc_t) xdr_fhstatus,(char*) &fhStatus,total_timeout);
if (fhStatus.fhs_status==0) {
atLeastOnceSucceeded=TRUE;
NFSFileHandle fh;
fh=fhStatus.fhstatus_u.fhs_fhandle;
TQString fname;
if ( exportlist->ex_dir[0] == '/' )
fname = KIO::encodeFileName(exportlist->ex_dir + 1);
else
fname = KIO::encodeFileName(exportlist->ex_dir);
m_handleCache.insert(TQString("/")+fname,fh);
m_exportedDirs.append(fname);
// kdDebug() <<"appending file -"<<fname<<"- with FH: -"<<fhStatus.fhstatus_u.fhs_fhandle<<"-"<<endl;
}
}
if (!atLeastOnceSucceeded)
{
closeConnection();
error( ERR_COULD_NOT_AUTHENTICATE, m_currentHost.latin1());
return;
}
server_addr.sin_port = 0;
//now create the client for the nfs daemon
//first get rid of the old one
closeConnection();
m_sock = RPC_ANYSOCK;
m_client = clnttcp_create(&server_addr,NFSPROG,NFSVERS,&m_sock,0,0);
if (m_client == 0)
{
server_addr.sin_port = 0;
m_sock = RPC_ANYSOCK;
pertry_timeout.tv_sec = 3;
pertry_timeout.tv_usec = 0;
m_client = clntudp_create(&server_addr,NFSPROG, NFSVERS, pertry_timeout, &m_sock);
if (m_client==0)
{
clnt_pcreateerror(const_cast<char *>("NFS clntudp_create"));
error(ERR_COULD_NOT_CONNECT, m_currentHost.latin1());
return;
}
}
m_client->cl_auth = authunix_create(hostName.data(),geteuid(),getegid(),0,0);
connected();
kdDebug(7121)<<"openConnection succeeded"<<endl;
}
void NFSProtocol::listDir( const KURL& _url)
{
KURL url(_url);
TQString path( TQFile::encodeName(url.path()));
if (path.isEmpty())
{
url.setPath("/");
redirection(url);
finished();
return;
}
//open the connection
if (m_client==0) openConnection();
//it failed
if (m_client==0) return;
if (isRoot(path))
{
kdDebug(7121)<<"listing root"<<endl;
totalSize( m_exportedDirs.count());
//in this case we don't need to do a real listdir
UDSEntry entry;
for (TQStringList::Iterator it=m_exportedDirs.begin(); it!=m_exportedDirs.end(); it++)
{
UDSAtom atom;
entry.clear();
atom.m_uds = KIO::UDS_NAME;
atom.m_str = (*it);
kdDebug(7121)<<"listing "<<(*it)<<endl;
entry.append( atom );
createVirtualDirEntry(entry);
listEntry( entry, false);
}
listEntry( entry, true ); // ready
finished();
return;
}
TQStringList filesToList;
kdDebug(7121)<<"getting subdir -"<<path<<"-"<<endl;
stripTrailingSlash(path);
NFSFileHandle fh=getFileHandle(path);
//cerr<<"this is the fh: -"<<fh<<"-"<<endl;
if (fh.isInvalid())
{
error( ERR_DOES_NOT_EXIST, path);
return;
}
readdirargs listargs;
memset(&listargs,0,sizeof(listargs));
listargs.count=1024*16;
memcpy(listargs.dir.data,fh,NFS_FHSIZE);
readdirres listres;
do
{
memset(&listres,'\0',sizeof(listres));
int clnt_stat = clnt_call(m_client, NFSPROC_READDIR, (xdrproc_t) xdr_readdirargs, (char*)&listargs,
(xdrproc_t) xdr_readdirres, (char*)&listres,total_timeout);
if (!checkForError(clnt_stat,listres.status,path)) return;
for (entry *dirEntry=listres.readdirres_u.reply.entries;dirEntry!=0;dirEntry=dirEntry->nextentry)
{
if ((TQString(".")!=dirEntry->name) && (TQString("..")!=dirEntry->name))
filesToList.append(dirEntry->name);
}
} while (!listres.readdirres_u.reply.eof);
totalSize( filesToList.count());
UDSEntry entry;
//stat all files in filesToList
for (TQStringList::Iterator it=filesToList.begin(); it!=filesToList.end(); it++)
{
UDSAtom atom;
diropargs dirargs;
diropres dirres;
memcpy(dirargs.dir.data,fh,NFS_FHSIZE);
TQCString tmpStr=TQFile::encodeName(*it);
dirargs.name=tmpStr.data();
kdDebug(7121)<<"calling rpc: FH: -"<<fh<<"- with name -"<<dirargs.name<<"-"<<endl;
int clnt_stat= clnt_call(m_client, NFSPROC_LOOKUP,
(xdrproc_t) xdr_diropargs, (char*)&dirargs,
(xdrproc_t) xdr_diropres, (char*)&dirres,total_timeout);
if (!checkForError(clnt_stat,dirres.status,(*it))) return;
NFSFileHandle tmpFH;
tmpFH=dirres.diropres_u.diropres.file.data;
m_handleCache.insert(path+"/"+(*it),tmpFH);
entry.clear();
atom.m_uds = KIO::UDS_NAME;
atom.m_str = (*it);
entry.append( atom );
//is it a symlink ?
if (S_ISLNK(dirres.diropres_u.diropres.attributes.mode))
{
kdDebug(7121)<<"it's a symlink !"<<endl;
//cerr<<"fh: "<<tmpFH<<endl;
nfs_fh nfsFH;
memcpy(nfsFH.data,dirres.diropres_u.diropres.file.data,NFS_FHSIZE);
//get the link dest
readlinkres readLinkRes;
char nameBuf[NFS_MAXPATHLEN];
readLinkRes.readlinkres_u.data=nameBuf;
int clnt_stat=clnt_call(m_client, NFSPROC_READLINK,
(xdrproc_t) xdr_nfs_fh, (char*)&nfsFH,
(xdrproc_t) xdr_readlinkres, (char*)&readLinkRes,total_timeout);
if (!checkForError(clnt_stat,readLinkRes.status,(*it))) return;
kdDebug(7121)<<"link dest is -"<<readLinkRes.readlinkres_u.data<<"-"<<endl;
TQCString linkDest(readLinkRes.readlinkres_u.data);
atom.m_uds = KIO::UDS_LINK_DEST;
atom.m_str = linkDest;
entry.append( atom );
bool isValid=isValidLink(path,linkDest);
if (!isValid)
{
completeBadLinkUDSEntry(entry,dirres.diropres_u.diropres.attributes);
}
else
{
if (isAbsoluteLink(linkDest))
{
completeAbsoluteLinkUDSEntry(entry,linkDest);
}
else
{
tmpStr=TQDir::cleanDirPath(path+TQString("/")+TQString(linkDest)).latin1();
dirargs.name=tmpStr.data();
tmpFH=getFileHandle(tmpStr);
memcpy(dirargs.dir.data,tmpFH,NFS_FHSIZE);
attrstat attrAndStat;
kdDebug(7121)<<"calling rpc: FH: -"<<fh<<"- with name -"<<dirargs.name<<"-"<<endl;
clnt_stat = clnt_call(m_client, NFSPROC_GETATTR,
(xdrproc_t) xdr_diropargs, (char*)&dirargs,
(xdrproc_t) xdr_attrstat, (char*)&attrAndStat,total_timeout);
if (!checkForError(clnt_stat,attrAndStat.status,tmpStr)) return;
completeUDSEntry(entry,attrAndStat.attrstat_u.attributes);
}
}
}
else
completeUDSEntry(entry,dirres.diropres_u.diropres.attributes);
listEntry( entry, false);
}
listEntry( entry, true ); // ready
finished();
}
void NFSProtocol::stat( const KURL & url)
{
TQString path( TQFile::encodeName(url.path()));
stripTrailingSlash(path);
kdDebug(7121)<<"NFS::stat for -"<<path<<"-"<<endl;
TQString tmpPath=path;
if ((tmpPath.length()>1) && (tmpPath[0]=='/')) tmpPath=tmpPath.mid(1);
// We can't stat root, but we know it's a dir
if (isRoot(path) || isExportedDir(path))
{
UDSEntry entry;
UDSAtom atom;
atom.m_uds = KIO::UDS_NAME;
atom.m_str = path;
entry.append( atom );
createVirtualDirEntry(entry);
// no size
statEntry( entry );
finished();
kdDebug(7121)<<"succeeded"<<endl;
return;
}
NFSFileHandle fh=getFileHandle(path);
if (fh.isInvalid())
{
error(ERR_DOES_NOT_EXIST,path);
return;
}
diropargs dirargs;
attrstat attrAndStat;
memcpy(dirargs.dir.data,fh,NFS_FHSIZE);
TQCString tmpStr=TQFile::encodeName(path);
dirargs.name=tmpStr.data();
kdDebug(7121)<<"calling rpc: FH: -"<<fh<<"- with name -"<<dirargs.name<<"-"<<endl;
int clnt_stat = clnt_call(m_client, NFSPROC_GETATTR,
(xdrproc_t) xdr_diropargs, (char*)&dirargs,
(xdrproc_t) xdr_attrstat, (char*)&attrAndStat,total_timeout);
if (!checkForError(clnt_stat,attrAndStat.status,path)) return;
UDSEntry entry;
entry.clear();
UDSAtom atom;
TQString fileName, parentDir;
getLastPart(path, fileName, parentDir);
stripTrailingSlash(parentDir);
atom.m_uds = KIO::UDS_NAME;
atom.m_str = fileName;
entry.append( atom );
//is it a symlink ?
if (S_ISLNK(attrAndStat.attrstat_u.attributes.mode))
{
kdDebug(7121)<<"it's a symlink !"<<endl;
nfs_fh nfsFH;
memcpy(nfsFH.data,fh,NFS_FHSIZE);
//get the link dest
readlinkres readLinkRes;
char nameBuf[NFS_MAXPATHLEN];
readLinkRes.readlinkres_u.data=nameBuf;
int clnt_stat=clnt_call(m_client, NFSPROC_READLINK,
(xdrproc_t) xdr_nfs_fh, (char*)&nfsFH,
(xdrproc_t) xdr_readlinkres, (char*)&readLinkRes,total_timeout);
if (!checkForError(clnt_stat,readLinkRes.status,path)) return;
kdDebug(7121)<<"link dest is -"<<readLinkRes.readlinkres_u.data<<"-"<<endl;
TQCString linkDest(readLinkRes.readlinkres_u.data);
atom.m_uds = KIO::UDS_LINK_DEST;
atom.m_str = linkDest;
entry.append( atom );
bool isValid=isValidLink(parentDir,linkDest);
if (!isValid)
{
completeBadLinkUDSEntry(entry,attrAndStat.attrstat_u.attributes);
}
else
{
if (isAbsoluteLink(linkDest))
{
completeAbsoluteLinkUDSEntry(entry,linkDest);
}
else
{
tmpStr=TQDir::cleanDirPath(parentDir+TQString("/")+TQString(linkDest)).latin1();
diropargs dirargs;
dirargs.name=tmpStr.data();
NFSFileHandle tmpFH;
tmpFH=getFileHandle(tmpStr);
memcpy(dirargs.dir.data,tmpFH,NFS_FHSIZE);
kdDebug(7121)<<"calling rpc: FH: -"<<fh<<"- with name -"<<dirargs.name<<"-"<<endl;
clnt_stat = clnt_call(m_client, NFSPROC_GETATTR,
(xdrproc_t) xdr_diropargs, (char*)&dirargs,
(xdrproc_t) xdr_attrstat, (char*)&attrAndStat,total_timeout);
if (!checkForError(clnt_stat,attrAndStat.status,tmpStr)) return;
completeUDSEntry(entry,attrAndStat.attrstat_u.attributes);
}
}
}
else
completeUDSEntry(entry,attrAndStat.attrstat_u.attributes);
statEntry( entry );
finished();
}
void NFSProtocol::completeAbsoluteLinkUDSEntry(UDSEntry& entry, const TQCString& path)
{
//taken from file.cc
struct stat buff;
if ( ::stat( path.data(), &buff ) == -1 ) return;
UDSAtom atom;
atom.m_uds = KIO::UDS_FILE_TYPE;
atom.m_long = buff.st_mode & S_IFMT; // extract file type
entry.append( atom );
atom.m_uds = KIO::UDS_ACCESS;
atom.m_long = buff.st_mode & 07777; // extract permissions
entry.append( atom );
atom.m_uds = KIO::UDS_SIZE;
atom.m_long = buff.st_size;
entry.append( atom );
atom.m_uds = KIO::UDS_MODIFICATION_TIME;
atom.m_long = buff.st_mtime;
entry.append( atom );
atom.m_uds = KIO::UDS_USER;
uid_t uid = buff.st_uid;
TQString *temp = m_usercache.find( uid );
if ( !temp )
{
struct passwd *user = getpwuid( uid );
if ( user )
{
m_usercache.insert( uid, new TQString(TQString::tqfromLatin1(user->pw_name)) );
atom.m_str = user->pw_name;
}
else
atom.m_str = "???";
}
else
atom.m_str = *temp;
entry.append( atom );
atom.m_uds = KIO::UDS_GROUP;
gid_t gid = buff.st_gid;
temp = m_groupcache.find( gid );
if ( !temp )
{
struct group *grp = getgrgid( gid );
if ( grp )
{
m_groupcache.insert( gid, new TQString(TQString::tqfromLatin1(grp->gr_name)) );
atom.m_str = grp->gr_name;
}
else
atom.m_str = "???";
}
else
atom.m_str = *temp;
entry.append( atom );
atom.m_uds = KIO::UDS_ACCESS_TIME;
atom.m_long = buff.st_atime;
entry.append( atom );
atom.m_uds = KIO::UDS_CREATION_TIME;
atom.m_long = buff.st_ctime;
entry.append( atom );
}
void NFSProtocol::completeBadLinkUDSEntry(UDSEntry& entry, fattr& attributes)
{
// It is a link pointing to nowhere
completeUDSEntry(entry,attributes);
UDSAtom atom;
atom.m_uds = KIO::UDS_FILE_TYPE;
atom.m_long = S_IFMT - 1;
entry.append( atom );
atom.m_uds = KIO::UDS_ACCESS;
atom.m_long = S_IRWXU | S_IRWXG | S_IRWXO;
entry.append( atom );
atom.m_uds = KIO::UDS_SIZE;
atom.m_long = 0L;
entry.append( atom );
}
void NFSProtocol::completeUDSEntry(UDSEntry& entry, fattr& attributes)
{
UDSAtom atom;
atom.m_uds = KIO::UDS_SIZE;
atom.m_long = attributes.size;
entry.append(atom);
atom.m_uds = KIO::UDS_MODIFICATION_TIME;
atom.m_long = attributes.mtime.seconds;
entry.append( atom );
atom.m_uds = KIO::UDS_ACCESS_TIME;
atom.m_long = attributes.atime.seconds;
entry.append( atom );
atom.m_uds = KIO::UDS_CREATION_TIME;
atom.m_long = attributes.ctime.seconds;
entry.append( atom );
atom.m_uds = KIO::UDS_ACCESS;
atom.m_long = (attributes.mode & 07777);
entry.append( atom );
atom.m_uds = KIO::UDS_FILE_TYPE;
atom.m_long =attributes.mode & S_IFMT; // extract file type
entry.append( atom );
atom.m_uds = KIO::UDS_USER;
uid_t uid = attributes.uid;
TQString *temp = m_usercache.find( uid );
if ( !temp )
{
struct passwd *user = getpwuid( uid );
if ( user )
{
m_usercache.insert( uid, new TQString(user->pw_name) );
atom.m_str = user->pw_name;
}
else
atom.m_str = "???";
}
else
atom.m_str = *temp;
entry.append( atom );
atom.m_uds = KIO::UDS_GROUP;
gid_t gid = attributes.gid;
temp = m_groupcache.find( gid );
if ( !temp )
{
struct group *grp = getgrgid( gid );
if ( grp )
{
m_groupcache.insert( gid, new TQString(grp->gr_name) );
atom.m_str = grp->gr_name;
}
else
atom.m_str = "???";
}
else
atom.m_str = *temp;
entry.append( atom );
/* KIO::UDSEntry::ConstIterator it = entry.begin();
for( ; it != entry.end(); it++ ) {
switch ((*it).m_uds) {
case KIO::UDS_FILE_TYPE:
kdDebug(7121) << "File Type : " << (mode_t)((*it).m_long) << endl;
break;
case KIO::UDS_ACCESS:
kdDebug(7121) << "Access permissions : " << (mode_t)((*it).m_long) << endl;
break;
case KIO::UDS_USER:
kdDebug(7121) << "User : " << ((*it).m_str.ascii() ) << endl;
break;
case KIO::UDS_GROUP:
kdDebug(7121) << "Group : " << ((*it).m_str.ascii() ) << endl;
break;
case KIO::UDS_NAME:
kdDebug(7121) << "Name : " << ((*it).m_str.ascii() ) << endl;
//m_strText = decodeFileName( (*it).m_str );
break;
case KIO::UDS_URL:
kdDebug(7121) << "URL : " << ((*it).m_str.ascii() ) << endl;
break;
case KIO::UDS_MIME_TYPE:
kdDebug(7121) << "MimeType : " << ((*it).m_str.ascii() ) << endl;
break;
case KIO::UDS_LINK_DEST:
kdDebug(7121) << "LinkDest : " << ((*it).m_str.ascii() ) << endl;
break;
}
}*/
}
void NFSProtocol::setHost(const TQString& host, int /*port*/, const TQString& /*user*/, const TQString& /*pass*/)
{
kdDebug(7121)<<"setHost: -"<<host<<"-"<<endl;
if (host.isEmpty())
{
error(ERR_UNKNOWN_HOST,"");
return;
}
if (host==m_currentHost) return;
m_currentHost=host;
m_handleCache.clear();
m_exportedDirs.clear();
closeConnection();
}
void NFSProtocol::mkdir( const KURL& url, int permissions )
{
kdDebug(7121)<<"mkdir"<<endl;
TQString thePath( TQFile::encodeName(url.path()));
stripTrailingSlash(thePath);
TQString dirName, parentDir;
getLastPart(thePath, dirName, parentDir);
stripTrailingSlash(parentDir);
kdDebug(7121)<<"path: -"<<thePath<<"- dir: -"<<dirName<<"- parentDir: -"<<parentDir<<"-"<<endl;
if (isRoot(parentDir))
{
error(ERR_WRITE_ACCESS_DENIED,thePath);
return;
}
NFSFileHandle fh=getFileHandle(parentDir);
if (fh.isInvalid())
{
error(ERR_DOES_NOT_EXIST,thePath);
return;
}
createargs createArgs;
memcpy(createArgs.where.dir.data,fh,NFS_FHSIZE);
TQCString tmpName=TQFile::encodeName(dirName);
createArgs.where.name=tmpName.data();
if (permissions==-1) createArgs.attributes.mode=0755;
else createArgs.attributes.mode=permissions;
diropres dirres;
int clnt_stat = clnt_call(m_client, NFSPROC_MKDIR,
(xdrproc_t) xdr_createargs, (char*)&createArgs,
(xdrproc_t) xdr_diropres, (char*)&dirres,total_timeout);
if (!checkForError(clnt_stat,dirres.status,thePath)) return;
finished();
}
bool NFSProtocol::checkForError(int clientStat, int nfsStat, const TQString& text)
{
if (clientStat!=RPC_SUCCESS)
{
kdDebug(7121)<<"rpc error: "<<clientStat<<endl;
//does this mapping make sense ?
error(ERR_CONNECTION_BROKEN,i18n("An RPC error occurred."));
return FALSE;
}
if (nfsStat!=NFS_OK)
{
kdDebug(7121)<<"nfs error: "<<nfsStat<<endl;
switch (nfsStat)
{
case NFSERR_PERM:
error(ERR_ACCESS_DENIED,text);
break;
case NFSERR_NOENT:
error(ERR_DOES_NOT_EXIST,text);
break;
//does this mapping make sense ?
case NFSERR_IO:
error(ERR_INTERNAL_SERVER,text);
break;
//does this mapping make sense ?
case NFSERR_NXIO:
error(ERR_DOES_NOT_EXIST,text);
break;
case NFSERR_ACCES:
error(ERR_ACCESS_DENIED,text);
break;
case NFSERR_EXIST:
error(ERR_FILE_ALREADY_EXIST,text);
break;
//does this mapping make sense ?
case NFSERR_NODEV:
error(ERR_DOES_NOT_EXIST,text);
break;
case NFSERR_NOTDIR:
error(ERR_IS_FILE,text);
break;
case NFSERR_ISDIR:
error(ERR_IS_DIRECTORY,text);
break;
//does this mapping make sense ?
case NFSERR_FBIG:
error(ERR_INTERNAL_SERVER,text);
break;
//does this mapping make sense ?
case NFSERR_NOSPC:
error(ERR_INTERNAL_SERVER,i18n("No space left on device"));
break;
case NFSERR_ROFS:
error(ERR_COULD_NOT_WRITE,i18n("Read only file system"));
break;
case NFSERR_NAMETOOLONG:
error(ERR_INTERNAL_SERVER,i18n("Filename too long"));
break;
case NFSERR_NOTEMPTY:
error(ERR_COULD_NOT_RMDIR,text);
break;
//does this mapping make sense ?
case NFSERR_DQUOT:
error(ERR_INTERNAL_SERVER,i18n("Disk quota exceeded"));
break;
case NFSERR_STALE:
error(ERR_DOES_NOT_EXIST,text);
break;
default:
error(ERR_UNKNOWN,text);
break;
}
return FALSE;
}
return TRUE;
}
void NFSProtocol::del( const KURL& url, bool isfile)
{
TQString thePath( TQFile::encodeName(url.path()));
stripTrailingSlash(thePath);
TQString fileName, parentDir;
getLastPart(thePath, fileName, parentDir);
stripTrailingSlash(parentDir);
kdDebug(7121)<<"del(): path: -"<<thePath<<"- file -"<<fileName<<"- parentDir: -"<<parentDir<<"-"<<endl;
if (isRoot(parentDir))
{
error(ERR_ACCESS_DENIED,thePath);
return;
}
NFSFileHandle fh=getFileHandle(parentDir);
if (fh.isInvalid())
{
error(ERR_DOES_NOT_EXIST,thePath);
return;
}
if (isfile)
{
kdDebug(7121)<<"Deleting file "<<thePath<<endl;
diropargs dirOpArgs;
memcpy(dirOpArgs.dir.data,fh,NFS_FHSIZE);
TQCString tmpName=TQFile::encodeName(fileName);
dirOpArgs.name=tmpName.data();
nfsstat nfsStat;
int clnt_stat = clnt_call(m_client, NFSPROC_REMOVE,
(xdrproc_t) xdr_diropargs, (char*)&dirOpArgs,
(xdrproc_t) xdr_nfsstat, (char*)&nfsStat,total_timeout);
if (!checkForError(clnt_stat,nfsStat,thePath)) return;
kdDebug(7121)<<"removing "<<thePath<<" from cache"<<endl;
m_handleCache.remove(m_handleCache.find(thePath));
finished();
}
else
{
kdDebug(7121)<<"Deleting directory "<<thePath<<endl;
diropargs dirOpArgs;
memcpy(dirOpArgs.dir.data,fh,NFS_FHSIZE);
TQCString tmpName=TQFile::encodeName(fileName);
dirOpArgs.name=tmpName.data();
nfsstat nfsStat;
int clnt_stat = clnt_call(m_client, NFSPROC_RMDIR,
(xdrproc_t) xdr_diropargs, (char*)&dirOpArgs,
(xdrproc_t) xdr_nfsstat, (char*)&nfsStat,total_timeout);
if (!checkForError(clnt_stat,nfsStat,thePath)) return;
kdDebug(7121)<<"removing "<<thePath<<" from cache"<<endl;
m_handleCache.remove(m_handleCache.find(thePath));
finished();
}
}
void NFSProtocol::chmod( const KURL& url, int permissions )
{
TQString thePath( TQFile::encodeName(url.path()));
stripTrailingSlash(thePath);
kdDebug( 7121 ) << "chmod -"<< thePath << "-"<<endl;
if (isRoot(thePath) || isExportedDir(thePath))
{
error(ERR_ACCESS_DENIED,thePath);
return;
}
NFSFileHandle fh=getFileHandle(thePath);
if (fh.isInvalid())
{
error(ERR_DOES_NOT_EXIST,thePath);
return;
}
sattrargs sAttrArgs;
memcpy(sAttrArgs.file.data,fh,NFS_FHSIZE);
sAttrArgs.attributes.uid=(unsigned int)-1;
sAttrArgs.attributes.gid=(unsigned int)-1;
sAttrArgs.attributes.size=(unsigned int)-1;
sAttrArgs.attributes.atime.seconds=(unsigned int)-1;
sAttrArgs.attributes.atime.useconds=(unsigned int)-1;
sAttrArgs.attributes.mtime.seconds=(unsigned int)-1;
sAttrArgs.attributes.mtime.useconds=(unsigned int)-1;
sAttrArgs.attributes.mode=permissions;
nfsstat nfsStat;
int clnt_stat = clnt_call(m_client, NFSPROC_SETATTR,
(xdrproc_t) xdr_sattrargs, (char*)&sAttrArgs,
(xdrproc_t) xdr_nfsstat, (char*)&nfsStat,total_timeout);
if (!checkForError(clnt_stat,nfsStat,thePath)) return;
finished();
}
void NFSProtocol::get( const KURL& url )
{
TQString thePath( TQFile::encodeName(url.path()));
kdDebug(7121)<<"get() -"<<thePath<<"-"<<endl;
NFSFileHandle fh=getFileHandle(thePath);
if (fh.isInvalid())
{
error(ERR_DOES_NOT_EXIST,thePath);
return;
}
readargs readArgs;
memcpy(readArgs.file.data,fh,NFS_FHSIZE);
readArgs.offset=0;
readArgs.count=NFS_MAXDATA;
readArgs.totalcount=NFS_MAXDATA;
readres readRes;
int offset(0);
char buf[NFS_MAXDATA];
readRes.readres_u.reply.data.data_val=buf;
TQByteArray array;
do
{
int clnt_stat = clnt_call(m_client, NFSPROC_READ,
(xdrproc_t) xdr_readargs, (char*)&readArgs,
(xdrproc_t) xdr_readres, (char*)&readRes,total_timeout);
if (!checkForError(clnt_stat,readRes.status,thePath)) return;
if (readArgs.offset==0)
totalSize(readRes.readres_u.reply.attributes.size);
offset=readRes.readres_u.reply.data.data_len;
//kdDebug(7121)<<"read "<<offset<<" bytes"<<endl;
readArgs.offset+=offset;
if (offset>0)
{
array.setRawData(readRes.readres_u.reply.data.data_val, offset);
data( array );
array.resetRawData(readRes.readres_u.reply.data.data_val, offset);
processedSize(readArgs.offset);
}
} while (offset>0);
data( TQByteArray() );
finished();
}
//TODO the partial putting thing is not yet implemented
void NFSProtocol::put( const KURL& url, int _mode, bool _overwrite, bool /*_resume*/ )
{
TQString destPath( TQFile::encodeName(url.path()));
kdDebug( 7121 ) << "Put -" << destPath <<"-"<<endl;
/*TQString dest_part( dest_orig );
dest_part += ".part";*/
stripTrailingSlash(destPath);
TQString parentDir, fileName;
getLastPart(destPath,fileName, parentDir);
if (isRoot(parentDir))
{
error(ERR_WRITE_ACCESS_DENIED,destPath);
return;
}
NFSFileHandle destFH;
destFH=getFileHandle(destPath);
kdDebug(7121)<<"file handle for -"<<destPath<<"- is "<<destFH<<endl;
//the file exists and we don't want to overwrite
if ((!_overwrite) && (!destFH.isInvalid()))
{
error(ERR_FILE_ALREADY_EXIST,destPath);
return;
}
//TODO: is this correct ?
//we have to "create" the file anyway, no matter if it already
//exists or not
//if we don't create it new, written text will be, hmm, "inserted"
//in the existing file, i.e. a file could not become smaller, since
//write only overwrites or extends, but doesn't remove stuff from a file (aleXXX)
kdDebug(7121)<<"creating the file -"<<fileName<<"-"<<endl;
NFSFileHandle parentFH;
parentFH=getFileHandle(parentDir);
//cerr<<"fh for parent dir: "<<parentFH<<endl;
//the directory doesn't exist
if (parentFH.isInvalid())
{
kdDebug(7121)<<"parent directory -"<<parentDir<<"- does not exist"<<endl;
error(ERR_DOES_NOT_EXIST,parentDir);
return;
}
createargs createArgs;
memcpy(createArgs.where.dir.data,(const char*)parentFH,NFS_FHSIZE);
TQCString tmpName=TQFile::encodeName(fileName);
createArgs.where.name=tmpName.data();
//the mode is apparently ignored if the file already exists
if (_mode==-1) createArgs.attributes.mode=0644;
else createArgs.attributes.mode=_mode;
createArgs.attributes.uid=geteuid();
createArgs.attributes.gid=getegid();
//this is required, otherwise we are not able to write shorter files
createArgs.attributes.size=0;
//hmm, do we need something here ? I don't think so
createArgs.attributes.atime.seconds=(unsigned int)-1;
createArgs.attributes.atime.useconds=(unsigned int)-1;
createArgs.attributes.mtime.seconds=(unsigned int)-1;
createArgs.attributes.mtime.useconds=(unsigned int)-1;
diropres dirOpRes;
int clnt_stat = clnt_call(m_client, NFSPROC_CREATE,
(xdrproc_t) xdr_createargs, (char*)&createArgs,
(xdrproc_t) xdr_diropres, (char*)&dirOpRes,total_timeout);
if (!checkForError(clnt_stat,dirOpRes.status,fileName)) return;
//we created the file successfully
//destFH=getFileHandle(destPath);
destFH=dirOpRes.diropres_u.diropres.file.data;
kdDebug(7121)<<"file -"<<fileName<<"- in dir -"<<parentDir<<"- created successfully"<<endl;
//cerr<<"with fh "<<destFH<<endl;
//now we can put
int result;
// Loop until we got 0 (end of data)
writeargs writeArgs;
memcpy(writeArgs.file.data,(const char*)destFH,NFS_FHSIZE);
writeArgs.beginoffset=0;
writeArgs.totalcount=0;
writeArgs.offset=0;
attrstat attrStat;
int bytesWritten(0);
kdDebug(7121)<<"starting to put"<<endl;
do
{
TQByteArray buffer;
dataReq(); // Request for data
result = readData( buffer );
//kdDebug(7121)<<"received "<<result<<" bytes for putting"<<endl;
char * data=buffer.data();
int bytesToWrite=buffer.size();
int writeNow(0);
if (result > 0)
{
do
{
if (bytesToWrite>NFS_MAXDATA)
{
writeNow=NFS_MAXDATA;
}
else
{
writeNow=bytesToWrite;
};
writeArgs.data.data_val=data;
writeArgs.data.data_len=writeNow;
int clnt_stat = clnt_call(m_client, NFSPROC_WRITE,
(xdrproc_t) xdr_writeargs, (char*)&writeArgs,
(xdrproc_t) xdr_attrstat, (char*)&attrStat,total_timeout);
//kdDebug(7121)<<"written"<<endl;
if (!checkForError(clnt_stat,attrStat.status,fileName)) return;
bytesWritten+=writeNow;
writeArgs.offset=bytesWritten;
//adjust the pointer
data=data+writeNow;
//decrease the rest
bytesToWrite-=writeNow;
} while (bytesToWrite>0);
}
} while ( result > 0 );
finished();
}
void NFSProtocol::rename( const KURL &src, const KURL &dest, bool _overwrite )
{
TQString srcPath( TQFile::encodeName(src.path()));
TQString destPath( TQFile::encodeName(dest.path()));
stripTrailingSlash(srcPath);
stripTrailingSlash(destPath);
kdDebug(7121)<<"renaming -"<<srcPath<<"- to -"<<destPath<<"-"<<endl;
if (isRoot(srcPath) || isExportedDir(srcPath))
{
error(ERR_CANNOT_RENAME,srcPath);
return;
}
if (!_overwrite)
{
NFSFileHandle testFH;
testFH=getFileHandle(destPath);
if (!testFH.isInvalid())
{
error(ERR_FILE_ALREADY_EXIST,destPath);
return;
}
}
TQString srcFileName, srcParentDir, destFileName, destParentDir;
getLastPart(srcPath, srcFileName, srcParentDir);
NFSFileHandle srcFH=getFileHandle(srcParentDir);
if (srcFH.isInvalid())
{
error(ERR_DOES_NOT_EXIST,srcParentDir);
return;
}
renameargs renameArgs;
memcpy(renameArgs.from.dir.data,srcFH,NFS_FHSIZE);
TQCString tmpName=TQFile::encodeName(srcFileName);
renameArgs.from.name=tmpName.data();
getLastPart(destPath, destFileName, destParentDir);
NFSFileHandle destFH=getFileHandle(destParentDir);
if (destFH.isInvalid())
{
error(ERR_DOES_NOT_EXIST,destParentDir);
return;
}
memcpy(renameArgs.to.dir.data,destFH,NFS_FHSIZE);
TQCString tmpName2=TQFile::encodeName(destFileName);
renameArgs.to.name=tmpName2.data();
nfsstat nfsStat;
int clnt_stat = clnt_call(m_client, NFSPROC_RENAME,
(xdrproc_t) xdr_renameargs, (char*)&renameArgs,
(xdrproc_t) xdr_nfsstat, (char*)&nfsStat,total_timeout);
if (!checkForError(clnt_stat,nfsStat,destPath)) return;
finished();
}
void NFSProtocol::copy( const KURL &src, const KURL &dest, int _mode, bool _overwrite )
{
//prepare the source
TQString thePath( TQFile::encodeName(src.path()));
stripTrailingSlash(thePath);
kdDebug( 7121 ) << "Copy to -" << thePath <<"-"<<endl;
NFSFileHandle fh=getFileHandle(thePath);
if (fh.isInvalid())
{
error(ERR_DOES_NOT_EXIST,thePath);
return;
};
//create the destination
TQString destPath( TQFile::encodeName(dest.path()));
stripTrailingSlash(destPath);
TQString parentDir, fileName;
getLastPart(destPath,fileName, parentDir);
if (isRoot(parentDir))
{
error(ERR_ACCESS_DENIED,destPath);
return;
}
NFSFileHandle destFH;
destFH=getFileHandle(destPath);
kdDebug(7121)<<"file handle for -"<<destPath<<"- is "<<destFH<<endl;
//the file exists and we don't want to overwrite
if ((!_overwrite) && (!destFH.isInvalid()))
{
error(ERR_FILE_ALREADY_EXIST,destPath);
return;
}
//TODO: is this correct ?
//we have to "create" the file anyway, no matter if it already
//exists or not
//if we don't create it new, written text will be, hmm, "inserted"
//in the existing file, i.e. a file could not become smaller, since
//write only overwrites or extends, but doesn't remove stuff from a file
kdDebug(7121)<<"creating the file -"<<fileName<<"-"<<endl;
NFSFileHandle parentFH;
parentFH=getFileHandle(parentDir);
//the directory doesn't exist
if (parentFH.isInvalid())
{
kdDebug(7121)<<"parent directory -"<<parentDir<<"- does not exist"<<endl;
error(ERR_DOES_NOT_EXIST,parentDir);
return;
};
createargs createArgs;
memcpy(createArgs.where.dir.data,(const char*)parentFH,NFS_FHSIZE);
TQCString tmpName=TQFile::encodeName(fileName);
createArgs.where.name=tmpName.data();
if (_mode==-1) createArgs.attributes.mode=0644;
else createArgs.attributes.mode=_mode;
createArgs.attributes.uid=geteuid();
createArgs.attributes.gid=getegid();
createArgs.attributes.size=0;
createArgs.attributes.atime.seconds=(unsigned int)-1;
createArgs.attributes.atime.useconds=(unsigned int)-1;
createArgs.attributes.mtime.seconds=(unsigned int)-1;
createArgs.attributes.mtime.useconds=(unsigned int)-1;
diropres dirOpRes;
int clnt_stat = clnt_call(m_client, NFSPROC_CREATE,
(xdrproc_t) xdr_createargs, (char*)&createArgs,
(xdrproc_t) xdr_diropres, (char*)&dirOpRes,total_timeout);
if (!checkForError(clnt_stat,dirOpRes.status,destPath)) return;
//we created the file successfully
destFH=dirOpRes.diropres_u.diropres.file.data;
kdDebug(7121)<<"file -"<<fileName<<"- in dir -"<<parentDir<<"- created successfully"<<endl;
char buf[NFS_MAXDATA];
writeargs writeArgs;
memcpy(writeArgs.file.data,(const char*)destFH,NFS_FHSIZE);
writeArgs.beginoffset=0;
writeArgs.totalcount=0;
writeArgs.offset=0;
writeArgs.data.data_val=buf;
attrstat attrStat;
readargs readArgs;
memcpy(readArgs.file.data,fh,NFS_FHSIZE);
readArgs.offset=0;
readArgs.count=NFS_MAXDATA;
readArgs.totalcount=NFS_MAXDATA;
readres readRes;
readRes.readres_u.reply.data.data_val=buf;
int bytesRead(0);
do
{
//first read
int clnt_stat = clnt_call(m_client, NFSPROC_READ,
(xdrproc_t) xdr_readargs, (char*)&readArgs,
(xdrproc_t) xdr_readres, (char*)&readRes,total_timeout);
if (!checkForError(clnt_stat,readRes.status,thePath)) return;
if (readArgs.offset==0)
totalSize(readRes.readres_u.reply.attributes.size);
bytesRead=readRes.readres_u.reply.data.data_len;
//kdDebug(7121)<<"read "<<bytesRead<<" bytes"<<endl;
//then write
if (bytesRead>0)
{
readArgs.offset+=bytesRead;
writeArgs.data.data_len=bytesRead;
clnt_stat = clnt_call(m_client, NFSPROC_WRITE,
(xdrproc_t) xdr_writeargs, (char*)&writeArgs,
(xdrproc_t) xdr_attrstat, (char*)&attrStat,total_timeout);
//kdDebug(7121)<<"written"<<endl;
if (!checkForError(clnt_stat,attrStat.status,destPath)) return;
writeArgs.offset+=bytesRead;
}
} while (bytesRead>0);
finished();
}
//TODO why isn't this even called ?
void NFSProtocol::symlink( const TQString &target, const KURL &dest, bool )
{
kdDebug(7121)<<"symlinking "<<endl;
TQString destPath=dest.path();
stripTrailingSlash(destPath);
TQString parentDir, fileName;
getLastPart(destPath,fileName, parentDir);
kdDebug(7121)<<"symlinking "<<parentDir<<" "<<fileName<<" to "<<target<<endl;
NFSFileHandle fh=getFileHandle(parentDir);
if (fh.isInvalid())
{
error(ERR_DOES_NOT_EXIST,parentDir);
return;
}
if (isRoot(parentDir))
{
error(ERR_ACCESS_DENIED,destPath);
return;
}
kdDebug(7121)<<"tach"<<endl;
TQCString tmpStr=target.latin1();
symlinkargs symLinkArgs;
symLinkArgs.to=tmpStr.data();
memcpy(symLinkArgs.from.dir.data,(const char*)fh,NFS_FHSIZE);
TQCString tmpStr2=TQFile::encodeName(destPath);
symLinkArgs.from.name=tmpStr2.data();
nfsstat nfsStat;
int clnt_stat = clnt_call(m_client, NFSPROC_SYMLINK,
(xdrproc_t) xdr_symlinkargs, (char*)&symLinkArgs,
(xdrproc_t) xdr_nfsstat, (char*)&nfsStat,total_timeout);
if (!checkForError(clnt_stat,nfsStat,destPath)) return;
finished();
}
bool NFSProtocol::isValidLink(const TQString& parentDir, const TQString& linkDest)
{
kdDebug(7121)<<"isValidLink: parent: "<<parentDir<<" link: "<<linkDest<<endl;
if (linkDest.isEmpty()) return FALSE;
if (isAbsoluteLink(linkDest))
{
kdDebug(7121)<<"is an absolute link"<<endl;
return TQFile::exists(linkDest);
}
else
{
kdDebug(7121)<<"is a relative link"<<endl;
TQString absDest=parentDir+"/"+linkDest;
kdDebug(7121)<<"pointing abs to "<<absDest<<endl;
absDest=removeFirstPart(absDest);
kdDebug(7121)<<"removed first part "<<absDest<<endl;
absDest=TQDir::cleanDirPath(absDest);
kdDebug(7121)<<"simplified to "<<absDest<<endl;
if (absDest.find("../")==0)
return FALSE;
kdDebug(7121)<<"is inside the nfs tree"<<endl;
absDest=parentDir+"/"+linkDest;
absDest=TQDir::cleanDirPath(absDest);
kdDebug(7121)<<"getting file handle of "<<absDest<<endl;
NFSFileHandle fh=getFileHandle(absDest);
return (!fh.isInvalid());
}
return FALSE;
}