Blame src/History.cc

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
}