grive2/libgrive/src/drive/Resource.cc

255 lines
5.9 KiB
C++
Raw Normal View History

/*
grive: an GPL program to sync a local directory with Google Drive
Copyright (C) 2012 Wan Wai Ho
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 version 2
of the License.
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; if not, write to the Free Software
Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301, USA.
*/
2012-05-19 11:41:21 +04:00
#include "Resource.hh"
#include "CommonUri.hh"
2012-05-19 12:18:33 +04:00
#include "http/Download.hh"
#include "http/StringResponse.hh"
#include "http/XmlResponse.hh"
#include "util/Crypt.hh"
#include "util/Log.hh"
#include "util/OS.hh"
#include "xml/Node.hh"
#include "xml/NodeSet.hh"
2012-04-26 20:55:10 +04:00
#include <cassert>
2012-05-19 12:18:33 +04:00
#include <fstream>
2012-04-26 20:55:10 +04:00
// for debugging
#include <iostream>
namespace gr {
2012-05-19 11:41:21 +04:00
Resource::Resource( const xml::Node& entry ) :
m_entry ( entry ),
m_parent ( 0 )
{
}
2012-05-19 12:18:33 +04:00
Resource::Resource( const Entry& entry, Resource *parent ) :
m_entry ( entry ),
2012-05-19 12:18:33 +04:00
m_parent ( parent )
{
}
2012-05-19 11:41:21 +04:00
Resource::Resource(
2012-05-19 13:14:04 +04:00
const std::string& name,
const std::string& kind,
2012-04-26 20:55:10 +04:00
const std::string& href ) :
2012-05-19 13:14:04 +04:00
m_entry ( name, kind, href ),
2012-04-26 20:55:10 +04:00
m_parent( 0 )
{
}
2012-05-19 11:41:21 +04:00
void Resource::Update( const Entry& e )
2012-05-17 20:37:11 +04:00
{
m_entry = e ;
}
2012-05-19 11:41:21 +04:00
std::string Resource::SelfHref() const
{
2012-05-09 19:52:06 +04:00
return m_entry.SelfHref() ;
}
2012-05-19 13:14:04 +04:00
std::string Resource::Name() const
{
2012-05-19 13:14:04 +04:00
return IsFolder() ? m_entry.Title() : m_entry.Filename() ;
}
2012-05-19 11:41:21 +04:00
std::string Resource::ResourceID() const
2012-05-17 20:37:11 +04:00
{
return m_entry.ResourceID() ;
}
2012-05-19 11:41:21 +04:00
const Resource* Resource::Parent() const
2012-04-26 20:55:10 +04:00
{
return m_parent ;
}
2012-05-19 11:41:21 +04:00
Resource* Resource::Parent()
{
return m_parent ;
}
2012-05-19 11:41:21 +04:00
std::string Resource::ParentHref() const
2012-05-09 19:52:06 +04:00
{
return m_entry.ParentHref() ;
}
2012-05-19 11:41:21 +04:00
void Resource::AddChild( Resource *child )
{
2012-04-26 20:55:10 +04:00
assert( child != 0 ) ;
assert( child->m_parent == 0 ) ;
assert( child != this ) ;
child->m_parent = this ;
m_child.push_back( child ) ;
}
2012-05-19 11:41:21 +04:00
void Resource::Swap( Resource& coll )
{
2012-05-09 19:52:06 +04:00
m_entry.Swap( coll.m_entry ) ;
2012-04-26 20:55:10 +04:00
std::swap( m_parent, coll.m_parent ) ;
m_child.swap( coll.m_child ) ;
}
2012-05-19 12:18:33 +04:00
bool Resource::IsFolder() const
{
return m_entry.Kind() == "folder" ;
}
fs::path Resource::Path() const
2012-04-26 20:55:10 +04:00
{
assert( m_parent != this ) ;
2012-05-19 12:18:33 +04:00
2012-05-19 13:14:04 +04:00
return m_parent != 0 ? (m_parent->Path() / Name()) : "." ;
2012-04-26 20:55:10 +04:00
}
2012-05-19 11:41:21 +04:00
bool Resource::IsInRootTree() const
{
return m_parent == 0 ? (SelfHref() == root_href) : m_parent->IsInRootTree() ;
}
2012-05-19 13:14:04 +04:00
Resource* Resource::FindChild( const std::string& name )
2012-05-17 20:37:11 +04:00
{
2012-05-19 11:41:21 +04:00
for ( std::vector<Resource*>::iterator i = m_child.begin() ; i != m_child.end() ; ++i )
2012-05-17 20:37:11 +04:00
{
assert( (*i)->m_parent == this ) ;
2012-05-19 13:14:04 +04:00
if ( (*i)->Name() == name )
2012-05-17 20:37:11 +04:00
return *i ;
}
return 0 ;
}
2012-05-19 12:18:33 +04:00
void Resource::Update( http::Agent *http, const http::Headers& auth )
{
assert( m_parent != 0 ) ;
bool changed = true ;
fs::path path = Path() ;
Trace( "updating %1%", path ) ;
// compare checksum first if file exists
std::ifstream ifile( path.string().c_str(), std::ios::binary | std::ios::in ) ;
if ( ifile && m_entry.ServerMD5() == crypt::MD5(ifile.rdbuf()) )
changed = false ;
// if the checksum is different, file is changed and we need to update
if ( changed )
{
DateTime remote = m_entry.ServerModified() ;
DateTime local = ifile ? os::FileMTime( path ) : DateTime() ;
// remote file is newer, download file
if ( !ifile || remote > local )
Download( http, path, auth ) ;
else
{
// re-reading the file
ifile.seekg(0) ;
Upload( http, ifile.rdbuf(), auth ) ;
}
}
}
void Resource::Delete( http::Agent *http, const http::Headers& auth )
{
http::Headers hdr( auth ) ;
hdr.push_back( "If-Match: " + m_entry.ETag() ) ;
http::StringResponse str ;
http->Custom( "DELETE", feed_base + "/" + m_entry.ResourceID() + "?delete=true", &str, hdr ) ;
}
void Resource::Download( http::Agent* http, const fs::path& file, const http::Headers& auth ) const
{
Log( "Downloading %1%", file ) ;
http::Download dl( file.string(), http::Download::NoChecksum() ) ;
long r = http->Get( m_entry.ContentSrc(), &dl, auth ) ;
if ( r <= 400 )
os::SetFileTime( file, m_entry.ServerModified() ) ;
}
bool Resource::Upload( http::Agent* http, std::streambuf *file, const http::Headers& auth )
{
// upload link missing means that file is read only
if ( m_entry.UploadLink().empty() )
{
Log( "Cannot upload %1%: file read-only.", m_entry.Title(), log::warning ) ;
return false ;
}
Log( "Uploading %1%", m_entry.Title() ) ;
std::string meta =
"<?xml version='1.0' encoding='UTF-8'?>\n"
"<entry xmlns=\"http://www.w3.org/2005/Atom\" xmlns:docs=\"http://schemas.google.com/docs/2007\">"
"<category scheme=\"http://schemas.google.com/g/2005#kind\" "
"term=\"http://schemas.google.com/docs/2007#file\"/>"
"<title>" + m_entry.Filename() + "</title>"
"</entry>" ;
std::string data(
(std::istreambuf_iterator<char>(file)),
(std::istreambuf_iterator<char>()) ) ;
std::ostringstream xcontent_len ;
xcontent_len << "X-Upload-Content-Length: " << data.size() ;
http::Headers hdr( auth ) ;
hdr.push_back( "Content-Type: application/atom+xml" ) ;
hdr.push_back( "X-Upload-Content-Type: application/octet-stream" ) ;
hdr.push_back( xcontent_len.str() ) ;
hdr.push_back( "If-Match: " + m_entry.ETag() ) ;
hdr.push_back( "Expect:" ) ;
http::StringResponse str ;
http->Put( m_entry.UploadLink(), meta, &str, hdr ) ;
std::string uplink = http->RedirLocation() ;
// parse the header and find "Location"
http::Headers uphdr ;
uphdr.push_back( "Expect:" ) ;
uphdr.push_back( "Accept:" ) ;
http::XmlResponse xml ;
http->Put( uplink, data, &xml, uphdr ) ;
Trace( "Receipted response = %1%", xml.Response() ) ;
m_entry.Update( xml.Response() ) ;
return true ;
}
} // end of namespace
namespace std
{
2012-05-19 11:41:21 +04:00
void swap( gr::Resource& c1, gr::Resource& c2 )
{
c1.Swap( c2 ) ;
}
}