/******************************************************************************* * Copyright (c) 2007, 2018 Association for Decentralized Information Management * in Industry THTH ry. * All rights reserved. This program and the accompanying materials * are made available under the terms of the Eclipse Public License v1.0 * which accompanies this distribution, and is available at * http://www.eclipse.org/legal/epl-v10.html * * Contributors: * VTT Technical Research Centre of Finland - initial API and implementation *******************************************************************************/ package org.simantics.db.impl.query; import java.util.ArrayList; import org.simantics.db.AsyncReadGraph; import org.simantics.db.exception.DatabaseException; import org.simantics.db.impl.BlockingAsyncMultiProcedure; import org.simantics.db.impl.graph.ReadGraphImpl; import org.simantics.db.procedure.AsyncMultiProcedure; import org.simantics.db.request.AsyncMultiRead; import org.simantics.db.request.RequestFlags; import org.slf4j.Logger; import org.slf4j.LoggerFactory; final public class AsyncMultiReadEntry extends CacheEntryBase> { private static final Logger LOGGER = LoggerFactory.getLogger(AsyncMultiReadEntry.class); protected AsyncMultiRead id; AsyncMultiReadEntry(AsyncMultiRead request) { this.id = request; } @Override int makeHash() { return id.hashCode(); } @Override public Object getOriginalRequest() { return id; } @Override public void discard() { super.discard(); id = null; setResult(null); } final synchronized public void finish(AsyncReadGraph graph) { assert(isPending()); synchronized(this) { setReady(); } } final synchronized public void except(AsyncReadGraph graph, Throwable t) { assert(isPending()); synchronized(this) { except(t); } } @SuppressWarnings("unchecked") final synchronized public void addOrSet(Object item) { assert(isPending()); ArrayList value = (ArrayList)getResult(); value.add((T)item); } @Override public void clearResult(QuerySupport support) { setResult(new ArrayList()); } @Override final public Query getQuery() { return new Query() { @Override public void recompute(ReadGraphImpl graph) { try { BlockingAsyncMultiProcedure proc = new BlockingAsyncMultiProcedure<>(graph.asyncBarrier, graph, new AsyncMultiProcedure() { @Override public void execute(AsyncReadGraph graph, T result) { addOrSet(result); } public void finished(AsyncReadGraph graph) { finish(graph); }; @Override public void exception(AsyncReadGraph graph, Throwable t) { except(t); } }, id); id.perform(graph , proc); proc.get(); } catch (Throwable t) { except(t); } } @Override public void removeEntry(QueryProcessor processor) { processor.cache.remove(AsyncMultiReadEntry.this); } @Override public int type() { return RequestFlags.INVALIDATE; } @Override public String toString() { if(id == null) return "DISCARDED"; else return id.toString() + statusOrException; } }; } @SuppressWarnings("unchecked") @Override public Object performFromCache(ReadGraphImpl graph, AsyncMultiProcedure proc) { if(isExcepted()) { try { proc.exception(graph, (Throwable)getResult()); } catch (Throwable t) { LOGGER.error("performFromCache proc.exception failed", t); } } else { final ArrayList values = (ArrayList)getResult(); for(T value : values) { try { proc.execute(graph, value); } catch (Throwable t) { LOGGER.error("performFromCache proc.execute failed", t); } } try { proc.finished(graph); } catch (Throwable t) { LOGGER.error("performFromCache proc.finished failed", t); } } return getResult(); } @Override public String toString() { if(id == null) return "DISCARDED"; else return id.toString() + statusOrException; } public Object compute(ReadGraphImpl graph, AsyncMultiProcedure procedure) throws DatabaseException { return graph.processor.cache.performQuery(graph, id, this, procedure); } }