|
Packit |
8f70b4 |
/*
|
|
Packit |
8f70b4 |
* lftp - file transfer program
|
|
Packit |
8f70b4 |
*
|
|
Packit |
8f70b4 |
* Copyright (c) 1996-2013 by Alexander V. Lukyanov (lav@yars.free.net)
|
|
Packit |
8f70b4 |
*
|
|
Packit |
8f70b4 |
* This program is free software; you can redistribute it and/or modify
|
|
Packit |
8f70b4 |
* it under the terms of the GNU General Public License as published by
|
|
Packit |
8f70b4 |
* the Free Software Foundation; either version 3 of the License, or
|
|
Packit |
8f70b4 |
* (at your option) any later version.
|
|
Packit |
8f70b4 |
*
|
|
Packit |
8f70b4 |
* This program is distributed in the hope that it will be useful,
|
|
Packit |
8f70b4 |
* but WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
Packit |
8f70b4 |
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
|
Packit |
8f70b4 |
* GNU General Public License for more details.
|
|
Packit |
8f70b4 |
*
|
|
Packit |
8f70b4 |
* You should have received a copy of the GNU General Public License
|
|
Packit |
8f70b4 |
* along with this program. If not, see <http://www.gnu.org/licenses/>.
|
|
Packit |
8f70b4 |
*/
|
|
Packit |
8f70b4 |
|
|
Packit |
8f70b4 |
#include <config.h>
|
|
Packit |
8f70b4 |
#include <sys/types.h>
|
|
Packit |
8f70b4 |
#include <sys/stat.h>
|
|
Packit |
8f70b4 |
#include <fcntl.h>
|
|
Packit |
8f70b4 |
#include <unistd.h>
|
|
Packit |
8f70b4 |
#include <stddef.h>
|
|
Packit |
8f70b4 |
#include "trio.h"
|
|
Packit |
8f70b4 |
#include "History.h"
|
|
Packit |
8f70b4 |
#include "url.h"
|
|
Packit |
8f70b4 |
#include "misc.h"
|
|
Packit |
8f70b4 |
|
|
Packit |
8f70b4 |
#define super KeyValueDB
|
|
Packit |
8f70b4 |
|
|
Packit |
8f70b4 |
History::History()
|
|
Packit |
8f70b4 |
{
|
|
Packit |
8f70b4 |
full=0;
|
|
Packit |
8f70b4 |
stamp=0;
|
|
Packit |
8f70b4 |
fd=-1;
|
|
Packit |
8f70b4 |
modified=false;
|
|
Packit |
8f70b4 |
const char *home=get_lftp_data_dir();
|
|
Packit |
8f70b4 |
if(home)
|
|
Packit |
8f70b4 |
file.vset(home,"/cwd_history",NULL);
|
|
Packit |
8f70b4 |
}
|
|
Packit |
8f70b4 |
|
|
Packit |
8f70b4 |
History::~History()
|
|
Packit |
8f70b4 |
{
|
|
Packit |
8f70b4 |
Close();
|
|
Packit |
8f70b4 |
if(full)
|
|
Packit |
8f70b4 |
delete full;
|
|
Packit |
8f70b4 |
}
|
|
Packit |
8f70b4 |
|
|
Packit |
8f70b4 |
const char *History::extract_url(const char *res)
|
|
Packit |
8f70b4 |
{
|
|
Packit |
8f70b4 |
const char *url=strchr(res,':');
|
|
Packit |
8f70b4 |
if(url)
|
|
Packit |
8f70b4 |
url++;
|
|
Packit |
8f70b4 |
else
|
|
Packit |
8f70b4 |
url=res;
|
|
Packit |
8f70b4 |
|
|
Packit |
8f70b4 |
if(url::is_url(url))
|
|
Packit |
8f70b4 |
return url;
|
|
Packit |
8f70b4 |
|
|
Packit |
8f70b4 |
return url::decode(url);
|
|
Packit |
8f70b4 |
}
|
|
Packit |
8f70b4 |
|
|
Packit |
8f70b4 |
time_t History::extract_stamp(const char *res)
|
|
Packit |
8f70b4 |
{
|
|
Packit |
8f70b4 |
return atol(res);
|
|
Packit |
8f70b4 |
}
|
|
Packit |
8f70b4 |
|
|
Packit |
8f70b4 |
const char *History::Lookup(const FileAccess *s)
|
|
Packit |
8f70b4 |
{
|
|
Packit |
8f70b4 |
const char *url=s->GetConnectURL(s->NO_PATH|s->NO_PASSWORD);
|
|
Packit |
8f70b4 |
if(!url)
|
|
Packit |
8f70b4 |
return 0;
|
|
Packit |
8f70b4 |
const char *res=super::Lookup(url);
|
|
Packit |
8f70b4 |
if(res)
|
|
Packit |
8f70b4 |
return extract_url(res);
|
|
Packit |
8f70b4 |
Refresh();
|
|
Packit |
8f70b4 |
Close();
|
|
Packit |
8f70b4 |
if(!full)
|
|
Packit |
8f70b4 |
return 0;
|
|
Packit |
8f70b4 |
res=full->Lookup(url);
|
|
Packit |
8f70b4 |
if(res)
|
|
Packit |
8f70b4 |
return extract_url(res);
|
|
Packit |
8f70b4 |
return 0;
|
|
Packit |
8f70b4 |
}
|
|
Packit |
8f70b4 |
|
|
Packit |
8f70b4 |
void History::Refresh()
|
|
Packit |
8f70b4 |
{
|
|
Packit |
8f70b4 |
if(!file)
|
|
Packit |
8f70b4 |
return;
|
|
Packit |
8f70b4 |
struct stat st;
|
|
Packit |
8f70b4 |
if((fd==-1 ? stat(file,&st) : fstat(fd,&st)) == -1)
|
|
Packit |
8f70b4 |
return;
|
|
Packit |
8f70b4 |
if(st.st_mtime==stamp)
|
|
Packit |
8f70b4 |
return;
|
|
Packit |
8f70b4 |
Load();
|
|
Packit |
8f70b4 |
}
|
|
Packit |
8f70b4 |
|
|
Packit |
8f70b4 |
void History::Load()
|
|
Packit |
8f70b4 |
{
|
|
Packit |
8f70b4 |
if(full)
|
|
Packit |
8f70b4 |
full->Empty();
|
|
Packit |
8f70b4 |
if(!file)
|
|
Packit |
8f70b4 |
return;
|
|
Packit |
8f70b4 |
if(fd==-1)
|
|
Packit |
8f70b4 |
{
|
|
Packit |
8f70b4 |
fd=open(file,O_RDONLY);
|
|
Packit |
8f70b4 |
if(fd==-1)
|
|
Packit |
8f70b4 |
return;
|
|
Packit |
8f70b4 |
fcntl(fd,F_SETFD,FD_CLOEXEC);
|
|
Packit |
8f70b4 |
if(Lock(fd,F_RDLCK)==-1)
|
|
Packit |
8f70b4 |
fprintf(stderr,"%s: lock for reading failed, trying to read anyway\n",file.get());
|
|
Packit |
8f70b4 |
}
|
|
Packit |
8f70b4 |
if(!full)
|
|
Packit |
8f70b4 |
full=new KeyValueDB;
|
|
Packit |
8f70b4 |
struct stat st;
|
|
Packit |
8f70b4 |
fstat(fd,&st);
|
|
Packit |
8f70b4 |
stamp=st.st_mtime;
|
|
Packit |
8f70b4 |
lseek(fd,0,SEEK_SET);
|
|
Packit |
8f70b4 |
|
|
Packit |
8f70b4 |
full->Read(dup(fd)); // Read closes fd
|
|
Packit |
8f70b4 |
}
|
|
Packit |
8f70b4 |
|
|
Packit |
8f70b4 |
void History::Close()
|
|
Packit |
8f70b4 |
{
|
|
Packit |
8f70b4 |
if(fd!=-1)
|
|
Packit |
8f70b4 |
{
|
|
Packit |
8f70b4 |
close(fd);
|
|
Packit |
8f70b4 |
fd=-1;
|
|
Packit |
8f70b4 |
}
|
|
Packit |
8f70b4 |
}
|
|
Packit |
8f70b4 |
|
|
Packit |
8f70b4 |
void History::Set(const FileAccess *s,const FileAccess::Path &cwd)
|
|
Packit |
8f70b4 |
{
|
|
Packit |
8f70b4 |
if(cwd.path==0 || !strcmp(cwd.path,"~") || s->GetHostName()==0)
|
|
Packit |
8f70b4 |
return;
|
|
Packit |
8f70b4 |
xstring res;
|
|
Packit |
8f70b4 |
res.setf("%lu:",(unsigned long)time(0));
|
|
Packit |
8f70b4 |
if(!cwd.url)
|
|
Packit |
8f70b4 |
{
|
|
Packit |
8f70b4 |
res.append_url_encoded(cwd,URL_PATH_UNSAFE);
|
|
Packit |
8f70b4 |
if(!cwd.is_file && url::dir_needs_trailing_slash(s->GetProto()) && res.last_char()!='/')
|
|
Packit |
8f70b4 |
res.append('/');
|
|
Packit |
8f70b4 |
}
|
|
Packit |
8f70b4 |
else
|
|
Packit |
8f70b4 |
res.append(cwd.url);
|
|
Packit |
8f70b4 |
super::Add(s->GetConnectURL(s->NO_PATH|s->NO_PASSWORD),res);
|
|
Packit |
8f70b4 |
modified=true;
|
|
Packit |
8f70b4 |
}
|
|
Packit |
8f70b4 |
|
|
Packit |
8f70b4 |
void History::Save()
|
|
Packit |
8f70b4 |
{
|
|
Packit |
8f70b4 |
Close();
|
|
Packit |
8f70b4 |
if(!file)
|
|
Packit |
8f70b4 |
return;
|
|
Packit |
8f70b4 |
if(!modified)
|
|
Packit |
8f70b4 |
return;
|
|
Packit |
8f70b4 |
fd=open(file,O_RDWR|O_CREAT,0600);
|
|
Packit |
8f70b4 |
if(fd==-1)
|
|
Packit |
8f70b4 |
return;
|
|
Packit |
8f70b4 |
fcntl(fd,F_SETFD,FD_CLOEXEC);
|
|
Packit |
8f70b4 |
if(Lock(fd,F_WRLCK)==-1)
|
|
Packit |
8f70b4 |
{
|
|
Packit |
8f70b4 |
fprintf(stderr,"%s: lock for writing failed\n",file.get());
|
|
Packit |
8f70b4 |
Close();
|
|
Packit |
8f70b4 |
return;
|
|
Packit |
8f70b4 |
}
|
|
Packit |
8f70b4 |
Refresh();
|
|
Packit |
8f70b4 |
|
|
Packit |
8f70b4 |
// merge
|
|
Packit |
8f70b4 |
int count=0;
|
|
Packit |
8f70b4 |
for(Pair *p=chain; p; p=p->next)
|
|
Packit |
8f70b4 |
{
|
|
Packit |
8f70b4 |
time_t new_stamp=extract_stamp(p->value);
|
|
Packit |
8f70b4 |
time_t old_stamp=0;
|
|
Packit |
8f70b4 |
const char *old_value=full->Lookup(p->key);
|
|
Packit |
8f70b4 |
if(old_value)
|
|
Packit |
8f70b4 |
old_stamp=extract_stamp(old_value);
|
|
Packit |
8f70b4 |
if(old_stamp
|
|
Packit |
8f70b4 |
{
|
|
Packit |
8f70b4 |
full->Add(p->key,p->value);
|
|
Packit |
8f70b4 |
count++;
|
|
Packit |
8f70b4 |
}
|
|
Packit |
8f70b4 |
}
|
|
Packit |
8f70b4 |
|
|
Packit |
8f70b4 |
if(count==0)
|
|
Packit |
8f70b4 |
{
|
|
Packit |
8f70b4 |
Close();
|
|
Packit |
8f70b4 |
return;
|
|
Packit |
8f70b4 |
}
|
|
Packit |
8f70b4 |
|
|
Packit |
8f70b4 |
lseek(fd,0,SEEK_SET);
|
|
Packit |
8f70b4 |
|
|
Packit |
8f70b4 |
#ifdef HAVE_FTRUNCATE
|
|
Packit |
8f70b4 |
if(ftruncate(fd,0)==-1) // note the following statement
|
|
Packit |
8f70b4 |
#endif
|
|
Packit |
8f70b4 |
close(open(file,O_WRONLY|O_TRUNC));
|
|
Packit |
8f70b4 |
|
|
Packit |
8f70b4 |
full->Write(fd);
|
|
Packit |
8f70b4 |
fd=-1; // Write closes file
|
|
Packit |
8f70b4 |
}
|