1
0
Fork 0
mirror of https://github.com/geometer/FBReaderJ.git synced 2025-10-05 10:49:24 +02:00
FBReaderJ/src/org/geometerplus/fbreader/network/NetworkLibrary.java
Nikolay Pultsin e8695cf6f2 formatting
2013-09-15 14:23:50 +01:00

683 lines
18 KiB
Java

/*
* Copyright (C) 2010-2013 Geometer Plus <contact@geometerplus.com>
*
* 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; either version 2 of the License, or
* (at your option) any later version.
*
* 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.
*/
package org.geometerplus.fbreader.network;
import java.util.*;
import java.lang.ref.WeakReference;
import org.geometerplus.zlibrary.core.library.ZLibrary;
import org.geometerplus.zlibrary.core.util.ZLNetworkUtil;
import org.geometerplus.zlibrary.core.util.MimeType;
import org.geometerplus.zlibrary.core.image.ZLImage;
import org.geometerplus.zlibrary.core.options.ZLBooleanOption;
import org.geometerplus.zlibrary.core.options.ZLStringListOption;
import org.geometerplus.zlibrary.core.options.ZLStringOption;
import org.geometerplus.zlibrary.core.network.ZLNetworkException;
import org.geometerplus.zlibrary.core.language.Language;
import org.geometerplus.zlibrary.core.resources.ZLResource;
import org.geometerplus.fbreader.tree.FBTree;
import org.geometerplus.fbreader.network.tree.*;
import org.geometerplus.fbreader.network.opds.OPDSLinkReader;
import org.geometerplus.fbreader.network.urlInfo.UrlInfo;
public class NetworkLibrary {
public interface ChangeListener {
public enum Code {
InitializationFinished,
InitializationFailed,
SomeCode,
/*
ItemAdded,
ItemRemoved,
StatusChanged,
*/
SignedIn,
Found,
NotFound,
EmptyCatalog,
NetworkError
}
void onLibraryChanged(Code code, Object[] params);
}
private static NetworkLibrary ourInstance;
public static NetworkLibrary Instance() {
if (ourInstance == null) {
ourInstance = new NetworkLibrary();
}
return ourInstance;
}
public static ZLResource resource() {
return ZLResource.resource("networkLibrary");
}
public interface OnNewLinkListener {
void onNewLink(INetworkLink link);
}
public final ZLStringOption NetworkSearchPatternOption =
new ZLStringOption("NetworkSearch", "Pattern", "");
// that's important to keep this list synchronized
// it can be used from background thread
private final List<INetworkLink> myLinks =
Collections.synchronizedList(new ArrayList<INetworkLink>());
private final Set<ChangeListener> myListeners =
Collections.synchronizedSet(new HashSet<ChangeListener>());
private final Map<NetworkTree,NetworkItemsLoader> myLoaders =
Collections.synchronizedMap(new HashMap<NetworkTree,NetworkItemsLoader>());
private final Map<String,WeakReference<ZLImage>> myImageMap =
Collections.synchronizedMap(new HashMap<String,WeakReference<ZLImage>>());
public List<String> languageCodes() {
final TreeSet<String> languageSet = new TreeSet<String>();
synchronized (myLinks) {
for (INetworkLink link : myLinks) {
languageSet.add(link.getLanguage());
}
}
return new ArrayList<String>(languageSet);
}
private ZLStringListOption myActiveLanguageCodesOption;
private ZLStringListOption activeLanguageCodesOption() {
if (myActiveLanguageCodesOption == null) {
myActiveLanguageCodesOption =
new ZLStringListOption(
"Options",
"ActiveLanguages",
ZLibrary.Instance().defaultLanguageCodes(),
","
);
}
return myActiveLanguageCodesOption;
}
public List<String> activeLanguageCodes() {
//if(isFirstLaunch){
// isFirstLaunch = false;
return activeLanguageCodesOption().getValue();
//}else{
// return new ArrayList<String>();
//}
}
public void setActiveLanguageCodes(Collection<String> codes) {
final TreeSet<String> allCodes = new TreeSet<String>();
allCodes.addAll(ZLibrary.Instance().defaultLanguageCodes());
allCodes.removeAll(languageCodes());
allCodes.addAll(codes);
// sort codes
final TreeSet<Language> languages = new TreeSet<Language>();
for (String code : allCodes) {
languages.add(new Language(code));
}
final ArrayList<String> codesList = new ArrayList<String>(languages.size());
for (Language l : languages) {
codesList.add(l.Code);
}
activeLanguageCodesOption().setValue(codesList);
invalidateChildren();
}
public List<String> linkIds() {
final TreeSet<String> idSet = new TreeSet<String>();
synchronized (myLinks) {
for (INetworkLink link : myLinks) {
idSet.add(link.getUrl(UrlInfo.Type.Catalog));
}
}
return new ArrayList<String>(idSet);
}
private ZLBooleanOption firstLaunchOption;
private ZLBooleanOption myFirstLaunchOption() {
if (firstLaunchOption == null) {
firstLaunchOption =
new ZLBooleanOption(
"Options",
"firstLaunch",
true
);
}
return firstLaunchOption;
}
private ZLStringListOption myActiveIdsOption;
private ZLStringListOption activeIdsOption() {
if (myActiveIdsOption == null) {
myActiveIdsOption =
new ZLStringListOption(
"Options",
"ActiveIds",
"",
","
);
}
return myActiveIdsOption;
}
public List<String> activeIds() {
return activeIdsOption().getValue();
}
public void setActiveOneId(String id){
List<String> ids = activeIdsOption().getValue();
final ArrayList<String> codesList = new ArrayList<String>();
codesList.addAll(ids);
codesList.add(id);
activeIdsOption().setValue(codesList);
invalidateChildren();
}
public void setActiveIds(Collection<String> ids) {
final TreeSet<String> allCodes = new TreeSet<String>();
allCodes.addAll(ids);
final ArrayList<String> codesList = new ArrayList<String>(allCodes.size());
for (String l : allCodes) {
codesList.add(l);
}
activeIdsOption().setValue(codesList);
invalidateChildren();
}
List<INetworkLink> activeLinks() {
final LinkedList<INetworkLink> filteredList = new LinkedList<INetworkLink>();
final Collection<String> ids = activeIds();
synchronized (myLinks) {
for (INetworkLink link : myLinks) {
if (ids.contains(link.getUrl(UrlInfo.Type.Catalog))) {
filteredList.add(link);
}
}
}
return filteredList;
}
public INetworkLink getLinkByUrl(String url) {
if (url == null) {
return null;
}
synchronized (myLinks) {
for (INetworkLink link : myLinks) {
if (url.equals(link.getUrlInfo(UrlInfo.Type.Catalog).Url)) {
return link;
}
}
}
return null;
}
public NetworkTree getCatalogTreeByUrl(String url) {
for (FBTree tree : getRootTree().subTrees()) {
if (tree instanceof NetworkCatalogRootTree) {
final String cUrl =
((NetworkCatalogTree)tree).getLink().getUrlInfo(UrlInfo.Type.Catalog).Url;
if (url.equals(cUrl)) {
return (NetworkTree)tree;
}
}
}
return null;
}
public NetworkTree getCatalogTreeByUrlAll(String url) {
for (FBTree tree : getRootAllTree().subTrees()) {
if (tree instanceof NetworkCatalogRootTree) {
final String cUrl =
((NetworkCatalogTree)tree).getLink().getUrlInfo(UrlInfo.Type.Catalog).Url;
if (url.equals(cUrl)) {
return (NetworkTree)tree;
}
}
}
return null;
}
public INetworkLink getLinkBySiteName(String siteName) {
synchronized (myLinks) {
for (INetworkLink link : myLinks) {
if (siteName.equals(link.getSiteName())) {
return link;
}
}
}
return null;
}
private final RootTree myRootAllTree = new RootTree("@AllRoot", false);
private final RootTree myRootTree = new RootTree("@Root", false);
private final RootTree myFakeRootTree = new RootTree("@FakeRoot", true);
private boolean myChildrenAreInvalid = true;
private boolean myUpdateVisibility;
private volatile boolean myIsInitialized;
private final SearchItem mySearchItem = new AllCatalogsSearchItem();
private NetworkLibrary() {
}
public boolean isInitialized() {
return myIsInitialized;
}
public synchronized void initialize() {
if (myIsInitialized) {
return;
}
try {
myLinks.addAll(OPDSLinkReader.loadOPDSLinks(OPDSLinkReader.CacheMode.LOAD));
} catch (ZLNetworkException e) {
removeAllLoadedLinks();
fireModelChangedEvent(ChangeListener.Code.InitializationFailed, e.getMessage());
return;
}
final NetworkDatabase db = NetworkDatabase.Instance();
if (db != null) {
myLinks.addAll(db.listLinks());
}
synchronize();
myIsInitialized = true;
fireModelChangedEvent(ChangeListener.Code.InitializationFinished);
}
private void removeAllLoadedLinks() {
final LinkedList<INetworkLink> toRemove = new LinkedList<INetworkLink>();
synchronized (myLinks) {
for (INetworkLink link : myLinks) {
if (!(link instanceof ICustomNetworkLink)) {
toRemove.add(link);
}
}
}
myLinks.removeAll(toRemove);
}
/*private void testDate(ATOMDateConstruct date1, ATOMDateConstruct date2) {
String sign = " == ";
final int diff = date1.compareTo(date2);
if (diff > 0) {
sign = " > ";
} else if (diff < 0) {
sign = " < ";
}
Log.w("FBREADER", "" + date1 + sign + date2);
}*/
private volatile boolean myUpdateInProgress;
private Object myUpdateLock = new Object();
public void runBackgroundUpdate(final boolean force) {
if (!isInitialized()) {
return;
}
final Thread thread = new Thread(new Runnable() {
public void run() {
try {
myUpdateInProgress = true;
fireModelChangedEvent(ChangeListener.Code.SomeCode);
runBackgroundUpdateInternal(force);
} catch (ZLNetworkException e) {
fireModelChangedEvent(ChangeListener.Code.NetworkError, e.getMessage());
} finally {
myUpdateInProgress = false;
fireModelChangedEvent(ChangeListener.Code.SomeCode);
}
}
});
thread.setPriority(Thread.MIN_PRIORITY);
thread.start();
}
private void runBackgroundUpdateInternal(boolean force) throws ZLNetworkException {
synchronized (myUpdateLock) {
final OPDSLinkReader.CacheMode mode =
force ? OPDSLinkReader.CacheMode.CLEAR : OPDSLinkReader.CacheMode.UPDATE;
final List<INetworkLink> loadedLinks = OPDSLinkReader.loadOPDSLinks(mode);
if (!loadedLinks.isEmpty()) {
removeAllLoadedLinks();
myLinks.addAll(loadedLinks);
}
invalidateChildren();
// we create this copy to prevent long operations on synchronized list
final List<INetworkLink> linksCopy = new ArrayList<INetworkLink>(myLinks);
for (INetworkLink link : linksCopy) {
if (link.getType() == INetworkLink.Type.Custom) {
final ICustomNetworkLink customLink = (ICustomNetworkLink)link;
if (force || customLink.isObsolete(12 * 60 * 60 * 1000)) { // 12 hours
try {
customLink.reloadInfo(true, true);
NetworkDatabase.Instance().saveLink(customLink);
} catch (Throwable t) {
// ignore
}
}
}
}
synchronize();
}
}
public String rewriteUrl(String url, boolean externalUrl) {
final String host = ZLNetworkUtil.hostFromUrl(url).toLowerCase();
synchronized (myLinks) {
for (INetworkLink link : myLinks) {
if (host.contains(link.getSiteName())) {
url = link.rewriteUrl(url, externalUrl);
}
}
}
return url;
}
private void invalidateChildren() {
myChildrenAreInvalid = true;
}
public void invalidateVisibility() {
myUpdateVisibility = true;
}
private void makeUpToDateRootAll() {
myRootAllTree.clear();
synchronized (myLinks) {
for (INetworkLink link : myLinks) {
int index = 0;
for (FBTree t : myRootAllTree.subTrees()) {
final INetworkLink l = ((NetworkTree)t).getLink();
if (l != null && link.compareTo(l) <= 0) {
break;
}
++index;
}
new NetworkCatalogRootTree(myRootAllTree, link, index);
}
}
}
private void makeUpToDate() {
updateActiveIds();
final SortedSet<INetworkLink> linkSet = new TreeSet<INetworkLink>(activeLinks());
final LinkedList<FBTree> toRemove = new LinkedList<FBTree>();
// we do remove sum tree items:
for (FBTree t : myRootTree.subTrees()) {
if (t instanceof NetworkCatalogTree) {
final INetworkLink link = ((NetworkCatalogTree)t).getLink();
if (link != null) {
if (!linkSet.contains(link)) {
// 1. links not listed in activeLinks list right now
toRemove.add(t);
} else if (link instanceof ICustomNetworkLink &&
((ICustomNetworkLink)link).hasChanges()) {
// 2. custom links that were changed
toRemove.add(t);
} else {
linkSet.remove(link);
}
} else {
// 3. search item
toRemove.add(t);
}
} else {
// 4. non-catalog nodes
toRemove.add(t);
}
}
for (FBTree tree : toRemove) {
tree.removeSelf();
}
// we do add new network catalog items
for (INetworkLink link : linkSet) {
int index = 0;
for (FBTree t : myRootTree.subTrees()) {
final INetworkLink l = ((NetworkTree)t).getLink();
if (l != null && link.compareTo(l) <= 0) {
break;
}
++index;
}
new NetworkCatalogRootTree(myRootTree, link, index);
}
// we do add non-catalog items
new SearchCatalogTree(myRootTree, mySearchItem, 0);
new AddCustomCatalogItemTree(myRootTree);
fireModelChangedEvent(ChangeListener.Code.SomeCode);
}
private void updateActiveIds(){
if(!myFirstLaunchOption().getValue()) return;
ArrayList<String> ids = new ArrayList<String>();
final Collection<String> codes = activeLanguageCodes();
synchronized (myLinks) {
for (INetworkLink link : myLinks) {
if (link instanceof ICustomNetworkLink ||
codes.contains(link.getLanguage())) {
ids.add(link.getUrl(UrlInfo.Type.Catalog));
}
}
}
setActiveIds(ids);
myFirstLaunchOption().setValue(false);
}
private void updateVisibility() {
for (FBTree tree : myRootTree.subTrees()) {
if (tree instanceof NetworkCatalogTree) {
((NetworkCatalogTree)tree).updateVisibility();
}
}
fireModelChangedEvent(ChangeListener.Code.SomeCode);
}
public void synchronize() {
if (myChildrenAreInvalid) {
myChildrenAreInvalid = false;
makeUpToDate();
makeUpToDateRootAll();
}
if (myUpdateVisibility) {
myUpdateVisibility = false;
updateVisibility();
}
}
public NetworkTree getRootTree() {
return myRootTree;
}
public NetworkTree getRootAllTree() {
return myRootAllTree;
}
public NetworkBookTree getFakeBookTree(NetworkBookItem book) {
final String id = book.getStringId();
for (FBTree tree : myFakeRootTree.subTrees()) {
if (tree instanceof NetworkBookTree &&
id.equals(tree.getUniqueKey().Id)) {
return (NetworkBookTree)tree;
}
}
return new NetworkBookTree(myFakeRootTree, book, true);
}
public BasketCatalogTree getFakeBasketTree(BasketItem item) {
final String id = item.getStringId();
for (FBTree tree : myFakeRootTree.subTrees()) {
if (tree instanceof BasketCatalogTree &&
id.equals(tree.getUniqueKey().Id)) {
return (BasketCatalogTree)tree;
}
}
return new BasketCatalogTree(myFakeRootTree, item);
}
public NetworkCatalogTree getFakeCatalogTree(NetworkCatalogItem item) {
final String id = item.getStringId();
for (FBTree tree : myFakeRootTree.subTrees()) {
if (tree instanceof NetworkCatalogTree &&
id.equals(tree.getUniqueKey().Id)) {
return (NetworkCatalogTree)tree;
}
}
return new NetworkCatalogTree(myFakeRootTree, item.Link, item, 0);
}
public NetworkTree getTreeByKey(NetworkTree.Key key) {
if (key == null) {
return null;
}
if (key.Parent == null) {
if (key.equals(myRootTree.getUniqueKey())) {
return myRootTree;
}
if (key.equals(myFakeRootTree.getUniqueKey())) {
return myFakeRootTree;
}
return null;
}
final NetworkTree parentTree = getTreeByKey(key.Parent);
if (parentTree == null) {
return null;
}
return parentTree != null ? (NetworkTree)parentTree.getSubTree(key.Id) : null;
}
public void addCustomLink(ICustomNetworkLink link) {
final int id = link.getId();
if (id == ICustomNetworkLink.INVALID_ID) {
myLinks.add(link);
} else {
synchronized (myLinks) {
for (int i = myLinks.size() - 1; i >= 0; --i) {
final INetworkLink l = myLinks.get(i);
if (l instanceof ICustomNetworkLink && ((ICustomNetworkLink)l).getId() == id) {
myLinks.set(i, link);
break;
}
}
}
}
NetworkDatabase.Instance().saveLink(link);
setActiveOneId(link.getUrl(UrlInfo.Type.Catalog));
invalidateChildren();
}
public void removeCustomLink(ICustomNetworkLink link) {
myLinks.remove(link);
NetworkDatabase.Instance().deleteLink(link);
invalidateChildren();
}
public void addChangeListener(ChangeListener listener) {
myListeners.add(listener);
}
public void removeChangeListener(ChangeListener listener) {
myListeners.remove(listener);
}
// TODO: change to private
/*private*/ public void fireModelChangedEvent(ChangeListener.Code code, Object ... params) {
synchronized (myListeners) {
for (ChangeListener l : myListeners) {
l.onLibraryChanged(code, params);
}
}
}
public final void storeLoader(NetworkTree tree, NetworkItemsLoader loader) {
myLoaders.put(tree, loader);
}
public final NetworkItemsLoader getStoredLoader(NetworkTree tree) {
return tree != null ? myLoaders.get(tree) : null;
}
public final boolean isUpdateInProgress() {
return myUpdateInProgress;
}
public final void startLoading(NetworkCatalogItem item) {
if (item != null) {
item.UpdatingInProgress = true;
fireModelChangedEvent(ChangeListener.Code.SomeCode);
}
}
public final void stopLoading(NetworkCatalogItem item) {
if (item != null) {
item.UpdatingInProgress = false;
fireModelChangedEvent(ChangeListener.Code.SomeCode);
}
}
public boolean isLoadingInProgress(NetworkTree tree) {
return
(tree instanceof NetworkCatalogTree &&
((NetworkCatalogTree)tree).Item.UpdatingInProgress) ||
getStoredLoader(tree) != null;
}
public final void removeStoredLoader(NetworkTree tree) {
myLoaders.remove(tree);
}
public ZLImage getImageByUrl(String url, MimeType mimeType) {
synchronized (myImageMap) {
WeakReference<ZLImage> ref = myImageMap.get(url);
if (ref != null) {
final ZLImage image = ref.get();
if (image != null) {
return image;
}
}
final ZLImage image = new NetworkImage(url, mimeType);
myImageMap.put(url, new WeakReference<ZLImage>(image));
return image;
}
}
}