/******************************************************************************* * 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 org.simantics.db.AsyncReadGraph; import org.simantics.db.exception.DatabaseException; import org.simantics.db.impl.graph.ReadGraphImpl; import org.simantics.db.procedure.AsyncProcedure; import org.simantics.db.request.Read; import org.simantics.db.request.ReadExt; import org.simantics.db.request.RequestFlags; import org.slf4j.Logger; import org.slf4j.LoggerFactory; public final class ReadEntry extends CacheEntryBase> implements AsyncProcedure { private static final Logger LOGGER = LoggerFactory.getLogger(ReadEntry.class); protected Read id; public ReadEntry(Read request) { this.id = request; } @Override int makeHash() { return id.hashCode(); } @Override public Object getOriginalRequest() { return id; } @Override public void discard() { super.discard(); setResult(null); } @Override final public Query getQuery() { return new Query() { @Override public void recompute(ReadGraphImpl graph) { try { T result = id.perform(graph); setResult(result); setReady(); } catch (Throwable t) { except(t); } } @Override public void removeEntry(QueryProcessor processor) { processor.cache.remove(ReadEntry.this); } @Override public int type() { if (id instanceof ReadExt) { return ((ReadExt) id).getType(); } else { return RequestFlags.INVALIDATE; } } @Override public String toString() { if (id == null) return "DISCARDED"; else return id.toString() + statusOrException; } }; } public static T computeForEach(ReadGraphImpl graph, Read request, ReadEntry entry, AsyncProcedure procedure_, boolean needsToBlock) throws DatabaseException { AsyncProcedure procedure = entry != null ? entry : procedure_; ReadGraphImpl queryGraph = graph.withParent(entry, null, needsToBlock); queryGraph.asyncBarrier.inc(); ReadGraphImpl executeGraph = graph.withParent(graph.parent, null, needsToBlock); executeGraph.asyncBarrier.inc(); try { // This throws T result = request.perform(queryGraph); if(procedure != null) procedure.execute(executeGraph, result); return (T)result; } catch (DatabaseException e) { if(procedure != null) procedure.exception(executeGraph, e); throw e; } catch (Throwable t) { DatabaseException dbe = new DatabaseException(t); if(procedure != null) procedure.exception(executeGraph, dbe); throw dbe; } finally { queryGraph.asyncBarrier.dec(); try { if (entry != null) { // This also throws so must dec barrier finally entry.performFromCache(executeGraph, procedure_); } } finally { executeGraph.asyncBarrier.dec(); if(needsToBlock) executeGraph.asyncBarrier.waitBarrier(procedure, executeGraph); } } } public Object performFromCache(ReadGraphImpl graph, AsyncProcedure procedure) throws DatabaseException { AsyncProcedure proc = (AsyncProcedure) procedure; if (isExcepted()) { if(proc != null) { try { proc.exception(graph, (Throwable) getResult()); } catch (Throwable t) { LOGGER.error("performFromCache proc.exception failed", t); } } Throwable t = (Throwable) getResult(); if(t instanceof DatabaseException) { throw (DatabaseException)t; } else { throw new DatabaseException(t); } } else { if(proc != null) { try { proc.execute(graph, (T) getResult()); } catch (Throwable t) { LOGGER.error("performFromCache proc.execute failed", t); } } return (T)getResult(); } } @Override public String toString() { if (id == null) return "DISCARDED"; else return id.toString() + " - " + statusOrException; } public Object get(ReadGraphImpl graph, AsyncProcedure procedure) throws DatabaseException { if (procedure != null) performFromCache(graph, procedure); checkAndThrow(); return getResult(); } @Override boolean isImmutable(ReadGraphImpl graph) throws DatabaseException { if (id instanceof ReadExt) { return ((ReadExt) id).isImmutable(graph); } return false; } @Override public void execute(AsyncReadGraph graph, T result) { setResult(result); setReady(); } @Override public void exception(AsyncReadGraph graph, Throwable throwable) { except(throwable); } }