Java tutorial
/** * Copyright (C) 2009 - present by OpenGamma Inc. and the OpenGamma group of companies * * Please see distribution for license. */ package com.opengamma.web.server; import java.util.ArrayList; import java.util.Collection; import java.util.Collections; import java.util.Comparator; import java.util.EnumSet; import java.util.HashMap; import java.util.List; import java.util.Map; import java.util.concurrent.ExecutorService; import java.util.regex.Pattern; import org.apache.commons.lang.StringUtils; import org.cometd.bayeux.server.BayeuxServer; import org.cometd.bayeux.server.BayeuxServer.SessionListener; import org.cometd.bayeux.server.ServerMessage; import org.cometd.bayeux.server.ServerSession; import org.cometd.server.AbstractService; import org.fudgemsg.FudgeContext; import org.slf4j.Logger; import org.slf4j.LoggerFactory; import org.threeten.bp.Instant; import org.threeten.bp.ZoneOffset; import org.threeten.bp.ZonedDateTime; import org.threeten.bp.format.DateTimeFormatter; import com.opengamma.DataNotFoundException; import com.opengamma.OpenGammaRuntimeException; import com.opengamma.core.change.ChangeEvent; import com.opengamma.core.change.ChangeListener; import com.opengamma.core.config.impl.ConfigItem; import com.opengamma.core.marketdatasnapshot.impl.ManageableMarketDataSnapshot; import com.opengamma.core.position.PositionSource; import com.opengamma.core.security.SecuritySource; import com.opengamma.engine.ComputationTargetResolver; import com.opengamma.engine.marketdata.NamedMarketDataSpecificationRepository; import com.opengamma.engine.marketdata.spec.MarketData; import com.opengamma.engine.marketdata.spec.MarketDataSpecification; import com.opengamma.engine.view.ViewDefinition; import com.opengamma.engine.view.ViewProcessor; import com.opengamma.engine.view.client.ViewClient; import com.opengamma.engine.view.execution.ExecutionFlags; import com.opengamma.engine.view.execution.ExecutionFlags.ParallelRecompilationMode; import com.opengamma.engine.view.execution.ExecutionOptions; import com.opengamma.engine.view.execution.ViewExecutionFlags; import com.opengamma.engine.view.execution.ViewExecutionOptions; import com.opengamma.financial.aggregation.PortfolioAggregationFunctions; import com.opengamma.id.UniqueId; import com.opengamma.id.VersionCorrection; import com.opengamma.livedata.UserPrincipal; import com.opengamma.master.config.ConfigMaster; import com.opengamma.master.marketdatasnapshot.MarketDataSnapshotDocument; import com.opengamma.master.marketdatasnapshot.MarketDataSnapshotHistoryRequest; import com.opengamma.master.marketdatasnapshot.MarketDataSnapshotHistoryResult; import com.opengamma.master.marketdatasnapshot.MarketDataSnapshotMaster; import com.opengamma.master.marketdatasnapshot.MarketDataSnapshotSearchRequest; import com.opengamma.master.marketdatasnapshot.impl.MarketDataSnapshotSearchIterator; import com.opengamma.master.portfolio.PortfolioMaster; import com.opengamma.master.position.PositionMaster; import com.opengamma.util.ArgumentChecker; import com.opengamma.web.server.conversion.ConversionMode; import com.opengamma.web.server.conversion.ResultConverterCache; /** * The core of the back-end to the web client, providing the implementation of the Bayeux protocol. */ @SuppressWarnings("deprecation") public class LiveResultsService extends AbstractService implements SessionListener { private static final Logger s_logger = LoggerFactory.getLogger(LiveResultsService.class); private static final DateTimeFormatter s_snapshotDateTimeFormatter = DateTimeFormatter .ofPattern("yyyy-MM-dd HH:mm:ss"); private static final Pattern s_guidPattern = Pattern.compile( "(\\{?([0-9a-fA-F]){8}-([0-9a-fA-F]){4}-([0-9a-fA-F]){4}-([0-9a-fA-F]){4}-([0-9a-fA-F]){12}\\}?)"); private final Map<String, WebView> _clientViews = new HashMap<String, WebView>(); /** * The executor service used to call web clients back asynchronously. */ private final ExecutorService _executorService; private final ViewProcessor _viewProcessor; private final MarketDataSnapshotMaster _snapshotMaster; private final UserPrincipal _user; private final ResultConverterCache _resultConverterCache; private final NamedMarketDataSpecificationRepository _namedMarketDataSpecificationRepository; private final AggregatedViewDefinitionManager _aggregatedViewDefinitionManager; private final ComputationTargetResolver _computationTargetResolver; public LiveResultsService(final BayeuxServer bayeux, final ViewProcessor viewProcessor, final PositionSource positionSource, final SecuritySource securitySource, final PortfolioMaster userPortfolioMaster, final PositionMaster userPositionMaster, final ConfigMaster userViewDefinitionRepository, final MarketDataSnapshotMaster snapshotMaster, final UserPrincipal user, final ExecutorService executorService, final FudgeContext fudgeContext, final NamedMarketDataSpecificationRepository namedMarketDataSpecificationRepository, final PortfolioAggregationFunctions portfolioAggregators, final ComputationTargetResolver computationTargetResolver) { super(bayeux, "processPortfolioRequest"); ArgumentChecker.notNull(bayeux, "bayeux"); ArgumentChecker.notNull(viewProcessor, "viewProcessor"); ArgumentChecker.notNull(positionSource, "positionSource"); ArgumentChecker.notNull(securitySource, "securitySource"); ArgumentChecker.notNull(userPortfolioMaster, "userPortfolioMaster"); ArgumentChecker.notNull(userPositionMaster, "userPositionMaster"); ArgumentChecker.notNull(userViewDefinitionRepository, "userViewDefinitionRepository"); ArgumentChecker.notNull(snapshotMaster, "snapshotMaster"); ArgumentChecker.notNull(user, "user"); ArgumentChecker.notNull(executorService, "executorService"); ArgumentChecker.notNull(namedMarketDataSpecificationRepository, "namedMarketDataSpecificationRepository"); ArgumentChecker.notNull(portfolioAggregators, "portfolioAggregators"); ArgumentChecker.notNull(computationTargetResolver, "computationTargetResolver"); _viewProcessor = viewProcessor; _snapshotMaster = snapshotMaster; _user = user; _executorService = executorService; _resultConverterCache = new ResultConverterCache(fudgeContext); _namedMarketDataSpecificationRepository = namedMarketDataSpecificationRepository; _aggregatedViewDefinitionManager = new AggregatedViewDefinitionManager(positionSource, securitySource, viewProcessor.getConfigSource(), userViewDefinitionRepository, userPortfolioMaster, userPositionMaster, portfolioAggregators.getMappedFunctions()); _computationTargetResolver = computationTargetResolver; viewProcessor.getConfigSource().changeManager().addChangeListener(new ChangeListener() { @Override public void entityChanged(ChangeEvent event) { sendInitData(false); } }); s_logger.info("Subscribing to services"); addService("/service/getInitData", "processInitDataRequest"); addService("/service/getSnapshotVersions", "processSnapshotVersionsRequest"); addService("/service/changeView", "processChangeViewRequest"); addService("/service/updates", "processUpdateRequest"); addService("/service/updates/mode", "processUpdateModeRequest"); addService("/service/updates/depgraph", "processDepGraphRequest"); addService("/service/currentview/pause", "processPauseRequest"); addService("/service/currentview/resume", "processResumeRequest"); getBayeux().addListener(this); s_logger.info("Finished subscribing to services"); } @Override public void sessionAdded(ServerSession session) { s_logger.debug("Session " + session.getId() + " connected"); } @Override public void sessionRemoved(ServerSession session, boolean timedout) { // Tidy up s_logger.debug("Session " + session.getId() + " disconnected"); if (_clientViews.containsKey(session.getId())) { WebView view = _clientViews.remove(session.getId()); shutDownWebView(view); } } public WebView getClientView(String clientId) { synchronized (_clientViews) { return _clientViews.get(clientId); } } private void initializeClientView(final ServerSession remote, final UniqueId baseViewDefinitionId, final String aggregatorName, final ViewExecutionOptions executionOptions, final UserPrincipal user) { synchronized (_clientViews) { WebView webView = _clientViews.get(remote.getId()); if (webView != null) { if (webView.matches(baseViewDefinitionId, aggregatorName, executionOptions)) { // Already initialized webView.reconnected(); return; } // Existing view is different - client is switching views shutDownWebView(webView); _clientViews.remove(remote.getId()); } ViewClient viewClient = getViewProcessor().createViewClient(user); try { UniqueId viewDefinitionId = _aggregatedViewDefinitionManager .getViewDefinitionId(baseViewDefinitionId, aggregatorName); webView = new WebView(getLocalSession(), remote, viewClient, baseViewDefinitionId, aggregatorName, viewDefinitionId, executionOptions, user, getExecutorService(), getResultConverterCache(), getComputationTargetResolver()); } catch (Exception e) { _aggregatedViewDefinitionManager.releaseViewDefinition(baseViewDefinitionId, aggregatorName); viewClient.shutdown(); throw new OpenGammaRuntimeException( "Error attaching client to view definition '" + baseViewDefinitionId + "'", e); } _clientViews.put(remote.getId(), webView); } } private void shutDownWebView(WebView webView) { webView.shutdown(); _aggregatedViewDefinitionManager.releaseViewDefinition(webView.getBaseViewDefinitionId(), webView.getAggregatorName()); } private UserPrincipal getUser(ServerSession remote) { return _user; } private ExecutorService getExecutorService() { return _executorService; } private ViewProcessor getViewProcessor() { return _viewProcessor; } private ResultConverterCache getResultConverterCache() { return _resultConverterCache; } private ComputationTargetResolver getComputationTargetResolver() { return _computationTargetResolver; } public void processUpdateRequest(ServerSession remote, ServerMessage message) { s_logger.info("Received portfolio data request from {}, getting client view...", remote); WebView webView = getClientView(remote.getId()); if (webView == null) { // Disconnected client has come back to life return; } webView.triggerUpdate(message); } @SuppressWarnings("unchecked") public void processUpdateModeRequest(ServerSession remote, ServerMessage message) { WebView webView = getClientView(remote.getId()); if (webView == null) { return; } Map<String, Object> dataMap = (Map<String, Object>) message.getData(); String gridName = (String) dataMap.get("gridName"); long jsRowId = (Long) dataMap.get("rowId"); long jsColId = (Long) dataMap.get("colId"); ConversionMode mode = ConversionMode.valueOf((String) dataMap.get("mode")); WebViewGrid grid = webView.getGridByName(gridName); if (grid == null) { s_logger.warn("Request to change update mode for cell in unknown grid '{}'", gridName); } grid.setConversionMode(WebGridCell.of((int) jsRowId, (int) jsColId), mode); } @SuppressWarnings("unchecked") public void processDepGraphRequest(ServerSession remote, ServerMessage message) { WebView webView = getClientView(remote.getId()); if (webView == null) { return; } Map<String, Object> dataMap = (Map<String, Object>) message.getData(); String gridName = (String) dataMap.get("gridName"); long jsRowId = (Long) dataMap.get("rowId"); long jsColId = (Long) dataMap.get("colId"); boolean includeDepGraph = (Boolean) dataMap.get("includeDepGraph"); webView.setIncludeDepGraph(gridName, WebGridCell.of((int) jsRowId, (int) jsColId), includeDepGraph); } public void processInitDataRequest(ServerSession remote, ServerMessage message) { s_logger.info("processInitDataRequest"); sendInitData(true); } private void sendInitData(boolean includeSnapshots) { Map<String, Object> reply = new HashMap<String, Object>(); Object availableViewDefinitions = getViewDefinitions(); reply.put("viewDefinitions", availableViewDefinitions); List<String> aggregatorNames = getAggregatorNames(); reply.put("aggregatorNames", aggregatorNames); if (includeSnapshots) { List<String> marketDataSpecificationNames = getMarketDataSpecificationNames(); reply.put("specifications", marketDataSpecificationNames); Map<String, Map<String, String>> snapshotDetails = getSnapshotDetails(); reply.put("snapshots", snapshotDetails); } getBayeux().createIfAbsent("/initData"); getBayeux().getChannel("/initData").publish(getLocalSession(), reply, null); } private List<Map<String, String>> getViewDefinitions() { List<Map<String, String>> result = new ArrayList<Map<String, String>>(); Collection<ConfigItem<ViewDefinition>> views = _viewProcessor.getConfigSource().getAll(ViewDefinition.class, VersionCorrection.LATEST); s_logger.debug("Available view entries: " + views); for (ConfigItem<ViewDefinition> view : views) { if (s_guidPattern.matcher(view.getName()).find()) { s_logger.debug("Ignoring view definition which appears to have an auto-generated name: {}", view.getName()); continue; } Map<String, String> resultEntry = new HashMap<String, String>(); resultEntry.put("id", view.getUniqueId().toLatest().toString()); resultEntry.put("name", view.getName()); result.add(resultEntry); } Collections.sort(result, new Comparator<Map<String, String>>() { @Override public int compare(Map<String, String> o1, Map<String, String> o2) { return o1.get("name").compareTo(o2.get("name")); } }); return result; } private List<String> getAggregatorNames() { List<String> result = new ArrayList<String>(); result.addAll(_aggregatedViewDefinitionManager.getAggregatorNames()); Collections.sort(result, String.CASE_INSENSITIVE_ORDER); return result; } private List<String> getMarketDataSpecificationNames() { return _namedMarketDataSpecificationRepository.getNames(); } private Map<String, Map<String, String>> getSnapshotDetails() { MarketDataSnapshotSearchRequest snapshotSearchRequest = new MarketDataSnapshotSearchRequest(); snapshotSearchRequest.setIncludeData(false); Map<String, Map<String, String>> snapshotsByBasisView = new HashMap<String, Map<String, String>>(); for (MarketDataSnapshotDocument doc : MarketDataSnapshotSearchIterator.iterable(_snapshotMaster, snapshotSearchRequest)) { ManageableMarketDataSnapshot snapshot = doc.getSnapshot(); if (snapshot.getUniqueId() == null) { s_logger.warn("Ignoring snapshot with null unique identifier {}", snapshot.getName()); continue; } if (StringUtils.isBlank(snapshot.getName())) { s_logger.warn("Ignoring snapshot {} with no name", snapshot.getUniqueId()); continue; } if (s_guidPattern.matcher(snapshot.getName()).find()) { s_logger.debug("Ignoring snapshot which appears to have an auto-generated name: {}", snapshot.getName()); continue; } String basisViewName = snapshot.getBasisViewName() != null ? snapshot.getBasisViewName() : "unknown"; Map<String, String> snapshotsForBasisView = snapshotsByBasisView.get(basisViewName); if (snapshotsForBasisView == null) { snapshotsForBasisView = new HashMap<String, String>(); snapshotsByBasisView.put(basisViewName, snapshotsForBasisView); } snapshotsForBasisView.put(snapshot.getUniqueId().getObjectId().toString(), snapshot.getName()); } return snapshotsByBasisView; } @SuppressWarnings("unchecked") public void processSnapshotVersionsRequest(ServerSession remote, ServerMessage message) { s_logger.info("processSnapshotVersionsRequest"); Map<String, Object> data = (Map<String, Object>) message.getData(); String snapshotIdString = (String) data.get("snapshotId"); UniqueId snapshotId = UniqueId.parse(snapshotIdString); MarketDataSnapshotHistoryRequest snapshotHistoryRequest = new MarketDataSnapshotHistoryRequest( snapshotId.getObjectId(), null, Instant.now()); snapshotHistoryRequest.setIncludeData(false); MarketDataSnapshotHistoryResult snapshotSearchResult = _snapshotMaster.history(snapshotHistoryRequest); String[][] versions = new String[snapshotSearchResult.getDocuments().size()][2]; int i = 0; for (MarketDataSnapshotDocument doc : snapshotSearchResult.getDocuments()) { ZonedDateTime snapshotDateTime = ZonedDateTime.ofInstant(doc.getVersionFromInstant(), ZoneOffset.UTC); versions[i][0] = doc.getUniqueId().toString(); versions[i][1] = snapshotDateTime.toString(s_snapshotDateTimeFormatter); i++; } Map<String, Object> response = new HashMap<String, Object>(); response.put("snapshotId", snapshotIdString); response.put("versions", versions); remote.deliver(getLocalSession(), "/snapshotVersions", response, null); } @SuppressWarnings("unchecked") public void processChangeViewRequest(ServerSession remote, ServerMessage message) { try { Map<String, Object> data = (Map<String, Object>) message.getData(); String viewIdString = (String) data.get("viewId"); UniqueId baseViewDefinitionId; try { baseViewDefinitionId = UniqueId.parse(viewIdString); } catch (IllegalArgumentException e) { sendChangeViewError(remote, "Invalid view definition identifier format: '" + viewIdString); return; } if (!validateViewDefinitionId(baseViewDefinitionId)) { sendChangeViewError(remote, "No view definition with identifier " + baseViewDefinitionId + " could be found"); return; } String aggregatorName = (String) data.get("aggregatorName"); String marketDataType = (String) data.get("marketDataType"); String versionDateTime = (String) data.get("versionDateTime"); MarketDataSpecification marketDataSpec; EnumSet<ViewExecutionFlags> flags; VersionCorrection versionCorrection; if (versionDateTime != null) { Instant versionAsOf = Instant.parse(versionDateTime); versionCorrection = VersionCorrection.ofVersionAsOf(versionAsOf); } else { versionCorrection = VersionCorrection.LATEST; } if ("snapshot".equals(marketDataType)) { String snapshotIdString = (String) data.get("snapshotId"); if (StringUtils.isBlank(snapshotIdString)) { sendChangeViewError(remote, "Unknown snapshot"); return; } UniqueId snapshotId = UniqueId.parse(snapshotIdString); if (snapshotId.isVersioned()) { if (versionCorrection != VersionCorrection.LATEST) { throw new OpenGammaRuntimeException( "Cannot specify both a versioned snapshot and a custom verion-correction"); } // If the version-correction is to be based on a snapshot then use the time at which the snapshot was created MarketDataSnapshotDocument snapshotDoc = _snapshotMaster.get(snapshotId.getObjectId(), versionCorrection); versionCorrection = VersionCorrection.ofVersionAsOf(snapshotDoc.getVersionFromInstant()); } else { try { MarketDataSnapshotDocument snapshotDoc = _snapshotMaster.get(snapshotId.getObjectId(), versionCorrection); snapshotId = snapshotDoc.getUniqueId(); } catch (DataNotFoundException e) { s_logger.error("Snapshot " + snapshotId.getObjectId() + " not found for version-correction " + versionCorrection, e); throw new OpenGammaRuntimeException( "selected snapshot not valid at chosen version date and time"); } } marketDataSpec = MarketData.user(snapshotId); flags = ExecutionFlags.none().triggerOnMarketData().get(); } else if ("live".equals(marketDataType)) { String liveMarketDataProvider = (String) data.get("provider"); marketDataSpec = _namedMarketDataSpecificationRepository.getSpecification(liveMarketDataProvider); flags = ExecutionFlags.triggersEnabled() .parallelCompilation(ParallelRecompilationMode.PARALLEL_EXECUTION).get(); } else { throw new OpenGammaRuntimeException("Unknown market data type: " + marketDataType); } ViewExecutionOptions executionOptions = ExecutionOptions.infinite(marketDataSpec, flags, versionCorrection); s_logger.info("Initializing view '{}', aggregated by '{}' with execution options '{}' for client '{}'", new Object[] { baseViewDefinitionId, aggregatorName, executionOptions, remote }); initializeClientView(remote, baseViewDefinitionId, aggregatorName, executionOptions, getUser(remote)); } catch (Exception e) { s_logger.error("Exception propagated to client while changing view", e); sendChangeViewError(remote, "Unexpected error with message: " + e.getMessage()); } } private void sendChangeViewError(ServerSession remote, String errorMessage) { s_logger.info("Notifying client of error changing view: " + errorMessage); Map<String, String> reply = new HashMap<String, String>(); reply.put("isError", "true"); reply.put("message", "Unable to change view. " + errorMessage + "."); remote.deliver(getLocalSession(), "/changeView", reply, null); } private boolean validateViewDefinitionId(UniqueId viewDefinitionId) { try { return _viewProcessor.getConfigSource().get(viewDefinitionId) != null; } catch (Exception e) { s_logger.warn("Error validating view definition ID " + viewDefinitionId, e); return false; } } public void processPauseRequest(ServerSession remote, ServerMessage message) { WebView webView = getClientView(remote.getId()); if (webView == null) { return; } webView.pause(); } public void processResumeRequest(ServerSession remote, ServerMessage message) { WebView webView = getClientView(remote.getId()); if (webView == null) { return; } webView.resume(); } }