/*************************************************************************** copyright : (C) 2003-2006 by Robby Stephenson email : robby@periapsis.org ***************************************************************************/ /*************************************************************************** * * * This program is free software; you can redistribute it and/or modify * * it under the terms of version 2 of the GNU General Public License as * * published by the Free Software Foundation; * * * ***************************************************************************/ #include #include "fetchmanager.h" #include "configwidget.h" #include "messagehandler.h" #include "../tellico_kernel.h" #include "../entry.h" #include "../collection.h" #include "../tellico_utils.h" #include "../tellico_debug.h" #ifdef AMAZON_SUPPORT #include "amazonfetcher.h" #endif #ifdef IMDB_SUPPORT #include "imdbfetcher.h" #endif #ifdef HAVE_YAZ #include "z3950fetcher.h" #endif #include "srufetcher.h" #include "entrezfetcher.h" #include "execexternalfetcher.h" #include "yahoofetcher.h" #include "animenfofetcher.h" #include "ibsfetcher.h" #include "isbndbfetcher.h" #include "gcstarpluginfetcher.h" #include "crossreffetcher.h" #include "arxivfetcher.h" #include "citebasefetcher.h" #include "bibsonomyfetcher.h" #include "googlescholarfetcher.h" #include "discogsfetcher.h" #include #include #include #include #include #include #include #include #include #include #include #define LOAD_ICON(name, group, size) \ KGlobal::iconLoader()->loadIcon(name, static_cast(group), size_) using Tellico::Fetch::Manager; Manager* Manager::s_self = 0; Manager::Manager() : TQObject(), m_currentFetcherIndex(-1), m_messager(new ManagerMessage()), m_count(0), m_loadDefaults(false) { loadFetchers(); // m_keyMap.insert(FetchFirst, TQString()); m_keyMap.insert(Title, i18n("Title")); m_keyMap.insert(Person, i18n("Person")); m_keyMap.insert(ISBN, i18n("ISBN")); m_keyMap.insert(UPC, i18n("UPC/EAN")); m_keyMap.insert(Keyword, i18n("Keyword")); m_keyMap.insert(DOI, i18n("DOI")); m_keyMap.insert(ArxivID, i18n("arXiv ID")); m_keyMap.insert(PubmedID, i18n("Pubmed ID")); // to keep from having a new i18n string, just remove octothorpe m_keyMap.insert(LCCN, i18n("LCCN#").remove('#')); m_keyMap.insert(Raw, i18n("Raw Query")); // m_keyMap.insert(FetchLast, TQString()); } Manager::~Manager() { delete m_messager; } void Manager::loadFetchers() { // myDebug() << "Manager::loadFetchers()" << endl; m_fetchers.clear(); m_configMap.clear(); KConfig* config = KGlobal::config(); if(config->hasGroup(TQString::tqfromLatin1("Data Sources"))) { KConfigGroup configGroup(config, TQString::tqfromLatin1("Data Sources")); int nSources = configGroup.readNumEntry("Sources Count", 0); for(int i = 0; i < nSources; ++i) { TQString group = TQString::tqfromLatin1("Data Source %1").tqarg(i); Fetcher::Ptr f = createFetcher(config, group); if(f) { m_configMap.insert(f, group); m_fetchers.append(f); f->setMessageHandler(m_messager); } } m_loadDefaults = false; } else { // add default sources m_fetchers = defaultFetchers(); m_loadDefaults = true; } } Tellico::Fetch::FetcherVec Manager::fetchers(int type_) { FetcherVec vec; for(FetcherVec::Iterator it = m_fetchers.begin(); it != m_fetchers.end(); ++it) { if(it->canFetch(type_)) { vec.append(it.data()); } } return vec; } Tellico::Fetch::KeyMap Manager::keyMap(const TQString& source_) const { // an empty string means return all if(source_.isEmpty()) { return m_keyMap; } // assume there's only one fetcher match KSharedPtr f = 0; for(FetcherVec::ConstIterator it = m_fetchers.constBegin(); it != m_fetchers.constEnd(); ++it) { if(source_ == it->source()) { f = it.data(); break; } } if(!f) { kdWarning() << "Manager::keyMap() - no fetcher found!" << endl; return KeyMap(); } KeyMap map; for(KeyMap::ConstIterator it = m_keyMap.begin(); it != m_keyMap.end(); ++it) { if(f->canSearch(it.key())) { map.insert(it.key(), it.data()); } } return map; } void Manager::startSearch(const TQString& source_, FetchKey key_, const TQString& value_) { if(value_.isEmpty()) { emit signalDone(); return; } // assume there's only one fetcher match int i = 0; m_currentFetcherIndex = -1; for(FetcherVec::Iterator it = m_fetchers.begin(); it != m_fetchers.end(); ++it, ++i) { if(source_ == it->source()) { ++m_count; // Fetcher::search() might emit done(), so increment before calling search() connect(it.data(), TQT_SIGNAL(signalResultFound(Tellico::Fetch::SearchResult*)), TQT_SIGNAL(signalResultFound(Tellico::Fetch::SearchResult*))); connect(it.data(), TQT_SIGNAL(signalDone(Tellico::Fetch::Fetcher::Ptr)), TQT_SLOT(slotFetcherDone(Tellico::Fetch::Fetcher::Ptr))); it->search(key_, value_); m_currentFetcherIndex = i; break; } } } void Manager::continueSearch() { if(m_currentFetcherIndex < 0 || m_currentFetcherIndex >= static_cast(m_fetchers.count())) { myDebug() << "Manager::continueSearch() - can't continue!" << endl; emit signalDone(); return; } Fetcher::Ptr f = m_fetchers[m_currentFetcherIndex]; if(f && f->hasMoreResults()) { ++m_count; connect(f, TQT_SIGNAL(signalResultFound(Tellico::Fetch::SearchResult*)), TQT_SIGNAL(signalResultFound(Tellico::Fetch::SearchResult*))); connect(f, TQT_SIGNAL(signalDone(Tellico::Fetch::Fetcher::Ptr)), TQT_SLOT(slotFetcherDone(Tellico::Fetch::Fetcher::Ptr))); f->continueSearch(); } else { emit signalDone(); } } bool Manager::hasMoreResults() const { if(m_currentFetcherIndex < 0 || m_currentFetcherIndex >= static_cast(m_fetchers.count())) { return false; } Fetcher::Ptr f = m_fetchers[m_currentFetcherIndex]; return f && f->hasMoreResults(); } void Manager::stop() { // myDebug() << "Manager::stop()" << endl; for(FetcherVec::Iterator it = m_fetchers.begin(); it != m_fetchers.end(); ++it) { if(it->isSearching()) { it->stop(); } } #ifndef NDEBUG if(m_count != 0) { myDebug() << "Manager::stop() - count should be 0!" << endl; } #endif m_count = 0; } void Manager::slotFetcherDone(Fetcher::Ptr fetcher_) { // myDebug() << "Manager::slotFetcherDone() - " << (fetcher_ ? fetcher_->source() : TQString()) // << " :" << m_count << endl; fetcher_->disconnect(); // disconnect all signals --m_count; if(m_count <= 0) { emit signalDone(); } } bool Manager::canFetch() const { for(FetcherVec::ConstIterator it = m_fetchers.constBegin(); it != m_fetchers.constEnd(); ++it) { if(it->canFetch(Kernel::self()->collectionType())) { return true; } } return false; } Tellico::Fetch::Fetcher::Ptr Manager::createFetcher(KConfig* config_, const TQString& group_) { if(!config_->hasGroup(group_)) { myDebug() << "Manager::createFetcher() - no config group for " << group_ << endl; return 0; } KConfigGroup config(config_, group_); int fetchType = config.readNumEntry("Type", Fetch::Unknown); if(fetchType == Fetch::Unknown) { myDebug() << "Manager::createFetcher() - unknown type " << fetchType << ", skipping" << endl; return 0; } Fetcher::Ptr f = 0; switch(fetchType) { case Amazon: #ifdef AMAZON_SUPPORT { int site = config.readNumEntry("Site", AmazonFetcher::Unknown); if(site == AmazonFetcher::Unknown) { myDebug() << "Manager::createFetcher() - unknown amazon site " << site << ", skipping" << endl; } else { f = new AmazonFetcher(static_cast(site), this); } } #endif break; case IMDB: #ifdef IMDB_SUPPORT f = new IMDBFetcher(this); #endif break; case Z3950: #ifdef HAVE_YAZ f = new Z3950Fetcher(this); #endif break; case SRU: f = new SRUFetcher(this); break; case Entrez: f = new EntrezFetcher(this); break; case ExecExternal: f = new ExecExternalFetcher(this); break; case Yahoo: f = new YahooFetcher(this); break; case AnimeNfo: f = new AnimeNfoFetcher(this); break; case IBS: f = new IBSFetcher(this); break; case ISBNdb: f = new ISBNdbFetcher(this); break; case GCstarPlugin: f = new GCstarPluginFetcher(this); break; case CrossRef: f = new CrossRefFetcher(this); break; case Arxiv: f = new ArxivFetcher(this); break; case Citebase: f = new CitebaseFetcher(this); break; case Bibsonomy: f = new BibsonomyFetcher(this); break; case GoogleScholar: f = new GoogleScholarFetcher(this); break; case Discogs: f = new DiscogsFetcher(this); break; case Unknown: default: break; } if(f) { f->readConfig(config, group_); } return f; } // static Tellico::Fetch::FetcherVec Manager::defaultFetchers() { FetcherVec vec; #ifdef AMAZON_SUPPORT vec.append(new AmazonFetcher(AmazonFetcher::US, this)); #endif #ifdef IMDB_SUPPORT vec.append(new IMDBFetcher(this)); #endif vec.append(SRUFetcher::libraryOfCongress(this)); vec.append(new ISBNdbFetcher(this)); vec.append(new YahooFetcher(this)); vec.append(new AnimeNfoFetcher(this)); vec.append(new ArxivFetcher(this)); vec.append(new GoogleScholarFetcher(this)); vec.append(new DiscogsFetcher(this)); // only add IBS if user includes italian if(KGlobal::locale()->languagesTwoAlpha().tqcontains(TQString::tqfromLatin1("it"))) { vec.append(new IBSFetcher(this)); } return vec; } Tellico::Fetch::FetcherVec Manager::createUpdateFetchers(int collType_) { if(m_loadDefaults) { return defaultFetchers(); } FetcherVec vec; KConfigGroup config(KGlobal::config(), "Data Sources"); int nSources = config.readNumEntry("Sources Count", 0); for(int i = 0; i < nSources; ++i) { TQString group = TQString::tqfromLatin1("Data Source %1").tqarg(i); // needs the KConfig* Fetcher::Ptr f = createFetcher(KGlobal::config(), group); if(f && f->canFetch(collType_) && f->canUpdate()) { vec.append(f); } } return vec; } Tellico::Fetch::FetcherVec Manager::createUpdateFetchers(int collType_, FetchKey key_) { FetcherVec fetchers; // creates new fetchers FetcherVec allFetchers = createUpdateFetchers(collType_); for(Fetch::FetcherVec::Iterator it = allFetchers.begin(); it != allFetchers.end(); ++it) { if(it->canSearch(key_)) { fetchers.append(it); } } return fetchers; } Tellico::Fetch::Fetcher::Ptr Manager::createUpdateFetcher(int collType_, const TQString& source_) { Fetcher::Ptr fetcher = 0; // creates new fetchers FetcherVec fetchers = createUpdateFetchers(collType_); for(Fetch::FetcherVec::Iterator it = fetchers.begin(); it != fetchers.end(); ++it) { if(it->source() == source_) { fetcher = it; break; } } return fetcher; } void Manager::updatetqStatus(const TQString& message_) { emit signaltqStatus(message_); } Tellico::Fetch::TypePairList Manager::typeList() { Fetch::TypePairList list; #ifdef AMAZON_SUPPORT list.append(TypePair(AmazonFetcher::defaultName(), Amazon)); #endif #ifdef IMDB_SUPPORT list.append(TypePair(IMDBFetcher::defaultName(), IMDB)); #endif #ifdef HAVE_YAZ list.append(TypePair(Z3950Fetcher::defaultName(), Z3950)); #endif list.append(TypePair(SRUFetcher::defaultName(), SRU)); list.append(TypePair(EntrezFetcher::defaultName(), Entrez)); list.append(TypePair(ExecExternalFetcher::defaultName(), ExecExternal)); list.append(TypePair(YahooFetcher::defaultName(), Yahoo)); list.append(TypePair(AnimeNfoFetcher::defaultName(), AnimeNfo)); list.append(TypePair(IBSFetcher::defaultName(), IBS)); list.append(TypePair(ISBNdbFetcher::defaultName(), ISBNdb)); list.append(TypePair(GCstarPluginFetcher::defaultName(), GCstarPlugin)); list.append(TypePair(CrossRefFetcher::defaultName(), CrossRef)); list.append(TypePair(ArxivFetcher::defaultName(), Arxiv)); list.append(TypePair(CitebaseFetcher::defaultName(), Citebase)); list.append(TypePair(BibsonomyFetcher::defaultName(), Bibsonomy)); list.append(TypePair(GoogleScholarFetcher::defaultName(),GoogleScholar)); list.append(TypePair(DiscogsFetcher::defaultName(), Discogs)); // now find all the scripts distributed with tellico TQStringList files = KGlobal::dirs()->findAllResources("appdata", TQString::tqfromLatin1("data-sources/*.spec"), false, true); for(TQStringList::Iterator it = files.begin(); it != files.end(); ++it) { KConfig spec(*it, false, false); TQString name = spec.readEntry("Name"); if(name.isEmpty()) { myDebug() << "Fetch::Manager::typeList() - no Name for " << *it << endl; continue; } if(!bundledScriptHasExecPath(*it, &spec)) { // no available exec continue; } list.append(TypePair(name, ExecExternal)); m_scriptMap.insert(name, *it); } list.sort(); return list; } // called when creating a new fetcher Tellico::Fetch::ConfigWidget* Manager::configWidget(TQWidget* tqparent_, Type type_, const TQString& name_) { ConfigWidget* w = 0; switch(type_) { #ifdef AMAZON_SUPPORT case Amazon: w = new AmazonFetcher::ConfigWidget(tqparent_); break; #endif #ifdef IMDB_SUPPORT case IMDB: w = new IMDBFetcher::ConfigWidget(tqparent_); break; #endif #ifdef HAVE_YAZ case Z3950: w = new Z3950Fetcher::ConfigWidget(tqparent_); break; #endif case SRU: w = new SRUConfigWidget(tqparent_); break; case Entrez: w = new EntrezFetcher::ConfigWidget(tqparent_); break; case ExecExternal: w = new ExecExternalFetcher::ConfigWidget(tqparent_); if(!name_.isEmpty() && m_scriptMap.tqcontains(name_)) { // bundledScriptHasExecPath() actually needs to write the exec path // back to the config so the configWidget can read it. But if the spec file // is not readablle, that doesn't work. So work around it with a copy to a temp file KTempFile tmpFile; tmpFile.setAutoDelete(true); KURL from, to; from.setPath(m_scriptMap[name_]); to.setPath(tmpFile.name()); // have to overwrite since KTempFile already created it if(!KIO::NetAccess::file_copy(from, to, -1, true /*overwrite*/)) { myDebug() << KIO::NetAccess::lastErrorString() << endl; } KConfig spec(to.path(), false, false); // pass actual location of spec file if(name_ == spec.readEntry("Name") && bundledScriptHasExecPath(m_scriptMap[name_], &spec)) { static_cast(w)->readConfig(&spec); } else { kdWarning() << "Fetch::Manager::configWidget() - Can't read config file for " << to.path() << endl; } } break; case Yahoo: w = new YahooFetcher::ConfigWidget(tqparent_); break; case AnimeNfo: w = new AnimeNfoFetcher::ConfigWidget(tqparent_); break; case IBS: w = new IBSFetcher::ConfigWidget(tqparent_); break; case ISBNdb: w = new ISBNdbFetcher::ConfigWidget(tqparent_); break; case GCstarPlugin: w = new GCstarPluginFetcher::ConfigWidget(tqparent_); break; case CrossRef: w = new CrossRefFetcher::ConfigWidget(tqparent_); break; case Arxiv: w = new ArxivFetcher::ConfigWidget(tqparent_); break; case Citebase: w = new CitebaseFetcher::ConfigWidget(tqparent_); break; case Bibsonomy: w = new BibsonomyFetcher::ConfigWidget(tqparent_); break; case GoogleScholar: w = new GoogleScholarFetcher::ConfigWidget(tqparent_); break; case Discogs: w = new DiscogsFetcher::ConfigWidget(tqparent_); break; case Unknown: kdWarning() << "Fetch::Manager::configWidget() - no widget defined for type = " << type_ << endl; } return w; } // static TQString Manager::typeName(Fetch::Type type_) { switch(type_) { #ifdef AMAZON_SUPPORT case Amazon: return AmazonFetcher::defaultName(); #endif #ifdef IMDB_SUPPORT case IMDB: return IMDBFetcher::defaultName(); #endif #ifdef HAVE_YAZ case Z3950: return Z3950Fetcher::defaultName(); #endif case SRU: return SRUFetcher::defaultName(); case Entrez: return EntrezFetcher::defaultName(); case ExecExternal: return ExecExternalFetcher::defaultName(); case Yahoo: return YahooFetcher::defaultName(); case AnimeNfo: return AnimeNfoFetcher::defaultName(); case IBS: return IBSFetcher::defaultName(); case ISBNdb: return ISBNdbFetcher::defaultName(); case GCstarPlugin: return GCstarPluginFetcher::defaultName(); case CrossRef: return CrossRefFetcher::defaultName(); case Arxiv: return ArxivFetcher::defaultName(); case Citebase: return CitebaseFetcher::defaultName(); case Bibsonomy: return BibsonomyFetcher::defaultName(); case GoogleScholar: return GoogleScholarFetcher::defaultName(); case Discogs: return DiscogsFetcher::defaultName(); case Unknown: break; } myWarning() << "Manager::typeName() - none found for " << type_ << endl; return TQString(); } TQPixmap Manager::fetcherIcon(Fetch::Fetcher::CPtr fetcher_, int group_, int size_) { #ifdef HAVE_YAZ if(fetcher_->type() == Fetch::Z3950) { const Fetch::Z3950Fetcher* f = static_cast(fetcher_.data()); KURL u; u.setProtocol(TQString::tqfromLatin1("http")); u.setHost(f->host()); TQString icon = favIcon(u); if(u.isValid() && !icon.isEmpty()) { return LOAD_ICON(icon, group_, size_); } } else #endif if(fetcher_->type() == Fetch::ExecExternal) { const Fetch::ExecExternalFetcher* f = static_cast(fetcher_.data()); const TQString p = f->execPath(); KURL u; if(p.tqfind(TQString::tqfromLatin1("allocine")) > -1) { u = TQString::tqfromLatin1("http://www.allocine.fr"); } else if(p.tqfind(TQString::tqfromLatin1("ministerio_de_cultura")) > -1) { u = TQString::tqfromLatin1("http://www.mcu.es"); } else if(p.tqfind(TQString::tqfromLatin1("dark_horse_comics")) > -1) { u = TQString::tqfromLatin1("http://www.darkhorse.com"); } else if(p.tqfind(TQString::tqfromLatin1("boardgamegeek")) > -1) { u = TQString::tqfromLatin1("http://www.boardgamegeek.com"); } else if(f->source().tqfind(TQString::tqfromLatin1("amarok"), 0, false /*case-sensitive*/) > -1) { return LOAD_ICON(TQString::tqfromLatin1("amarok"), group_, size_); } if(!u.isEmpty() && u.isValid()) { TQString icon = favIcon(u); if(!icon.isEmpty()) { return LOAD_ICON(icon, group_, size_); } } } return fetcherIcon(fetcher_->type(), group_); } TQPixmap Manager::fetcherIcon(Fetch::Type type_, int group_, int size_) { TQString name; switch(type_) { case Amazon: name = favIcon("http://amazon.com"); break; case IMDB: name = favIcon("http://imdb.com"); break; case Z3950: name = TQString::tqfromLatin1("network"); break; // rather arbitrary case SRU: name = TQString::tqfromLatin1("network_local"); break; // just to be different than z3950 case Entrez: name = favIcon("http://www.ncbi.nlm.nih.gov"); break; case ExecExternal: name = TQString::tqfromLatin1("exec"); break; case Yahoo: name = favIcon("http://yahoo.com"); break; case AnimeNfo: name = favIcon("http://animenfo.com"); break; case IBS: name = favIcon("http://internetbookshop.it"); break; case ISBNdb: name = favIcon("http://isbndb.com"); break; case GCstarPlugin: name = TQString::tqfromLatin1("gcstar"); break; case CrossRef: name = favIcon("http://crossref.org"); break; case Arxiv: name = favIcon("http://arxiv.org"); break; case Citebase: name = favIcon("http://citebase.org"); break; case Bibsonomy: name = favIcon("http://bibsonomy.org"); break; case GoogleScholar: name = favIcon("http://scholar.google.com"); break; case Discogs: name = favIcon("http://www.discogs.com"); break; case Unknown: kdWarning() << "Fetch::Manager::fetcherIcon() - no pixmap defined for type = " << type_ << endl; } return name.isEmpty() ? TQPixmap() : LOAD_ICON(name, group_, size_); } TQString Manager::favIcon(const KURL& url_) { DCOPRef kded("kded", "favicons"); DCOPReply reply = kded.call("iconForURL(KURL)", url_); TQString iconName = reply.isValid() ? reply : TQString(); if(!iconName.isEmpty()) { return iconName; } else { // go ahead and try to download it for later kded.call("downloadHostIcon(KURL)", url_); } return KMimeType::iconForURL(url_); } bool Manager::bundledScriptHasExecPath(const TQString& specFile_, KConfig* config_) { // make sure ExecPath is set and executable // for the bundled scripts, either the exec name is not set, in which case it is the // name of the spec file, minus the .spec, or the exec is set, and is local to the dir // if not, look for it TQString exec = config_->readPathEntry("ExecPath"); TQFileInfo specInfo(specFile_), execInfo(exec); if(exec.isEmpty() || !execInfo.exists()) { exec = specInfo.dirPath(true) + TQDir::separator() + specInfo.baseName(true); // remove ".spec" } else if(execInfo.isRelative()) { exec = specInfo.dirPath(true) + exec; } else if(!execInfo.isExecutable()) { kdWarning() << "Fetch::Manager::execPathForBundledScript() - not executable: " << specFile_ << endl; return false; } execInfo.setFile(exec); if(!execInfo.exists() || !execInfo.isExecutable()) { kdWarning() << "Fetch::Manager::execPathForBundledScript() - no exec file for " << specFile_ << endl; kdWarning() << "exec = " << exec << endl; return false; // we're not ok } config_->writePathEntry("ExecPath", exec); config_->sync(); // might be readonly, but that's ok return true; } #include "fetchmanager.moc"