com.squid.kraken.v4.api.core.analytics.AnalyticsServiceBaseImpl.java Source code

Java tutorial

Introduction

Here is the source code for com.squid.kraken.v4.api.core.analytics.AnalyticsServiceBaseImpl.java

Source

/*******************************************************************************
 * Copyright  Squid Solutions, 2016
 *
 * This file is part of Open Bouquet software.
 *  
 * This program is free software; you can redistribute it and/or modify
 * it under the terms of the GNU Affero General Public License as
 * published by the Free Software Foundation (version 3 of the License).
 *
 * There is a special FOSS exception to the terms and conditions of the 
 * licenses as they are applied to this program. See LICENSE.txt in
 * the directory of this program distribution.
 *
 * 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.
 *
 * Squid Solutions also offers commercial licenses with additional warranties,
 * professional functionalities or services. If you purchase a commercial
 * license, then it supersedes and replaces any other agreement between
 * you and Squid Solutions (above licenses and LICENSE.txt included).
 * See http://www.squidsolutions.com/EnterpriseBouquet/
 *******************************************************************************/
package com.squid.kraken.v4.api.core.analytics;

import java.io.IOException;
import java.io.OutputStream;
import java.io.UnsupportedEncodingException;
import java.net.URI;
import java.net.URISyntaxException;
import java.net.URLEncoder;
import java.text.ParseException;
import java.text.SimpleDateFormat;
import java.util.ArrayList;
import java.util.Arrays;
import java.util.Collection;
import java.util.Collections;
import java.util.Comparator;
import java.util.Date;
import java.util.HashMap;
import java.util.HashSet;
import java.util.IllegalFormatException;
import java.util.List;
import java.util.Map.Entry;
import java.util.concurrent.Callable;
import java.util.concurrent.ExecutionException;
import java.util.concurrent.Future;
import java.util.concurrent.TimeoutException;
import java.util.zip.GZIPOutputStream;

import javax.ws.rs.WebApplicationException;
import javax.ws.rs.core.MediaType;
import javax.ws.rs.core.MultivaluedMap;
import javax.ws.rs.core.Response;
import javax.ws.rs.core.Response.ResponseBuilder;
import javax.ws.rs.core.StreamingOutput;
import javax.ws.rs.core.UriBuilder;
import javax.ws.rs.core.UriBuilderException;
import javax.ws.rs.core.UriInfo;

import org.apache.commons.codec.binary.Base64;
import org.apache.commons.lang3.StringEscapeUtils;
import org.apache.cxf.jaxrs.impl.UriBuilderImpl;
import org.slf4j.Logger;
import org.slf4j.LoggerFactory;

import com.fasterxml.jackson.annotation.JsonInclude.Include;
import com.fasterxml.jackson.core.JsonProcessingException;
import com.fasterxml.jackson.databind.DeserializationFeature;
import com.fasterxml.jackson.databind.JsonNode;
import com.fasterxml.jackson.databind.ObjectMapper;
import com.squid.core.concurrent.ExecutionManager;
import com.squid.core.database.impl.DatabaseServiceException;
import com.squid.core.domain.DomainNumericConstant;
import com.squid.core.domain.IDomain;
import com.squid.core.domain.operators.IntrinsicOperators;
import com.squid.core.domain.operators.OperatorDefinition;
import com.squid.core.domain.sort.DomainSort;
import com.squid.core.domain.sort.DomainSort.SortDirection;
import com.squid.core.domain.sort.SortOperatorDefinition;
import com.squid.core.domain.vector.VectorOperatorDefinition;
import com.squid.core.expression.ConstantValue;
import com.squid.core.expression.ExpressionAST;
import com.squid.core.expression.Operator;
import com.squid.core.expression.PrettyPrintOptions;
import com.squid.core.expression.PrettyPrintOptions.ReferenceStyle;
import com.squid.core.expression.scope.ExpressionMaker;
import com.squid.core.expression.scope.ScopeException;
import com.squid.core.poi.ExcelFile;
import com.squid.core.poi.ExcelSettingsBean;
import com.squid.core.sql.model.SQLScopeException;
import com.squid.core.sql.render.RenderingException;
import com.squid.kraken.v4.KrakenConfig;
import com.squid.kraken.v4.api.core.APIException;
import com.squid.kraken.v4.api.core.AccessRightsUtils;
import com.squid.kraken.v4.api.core.ComputingInProgressAPIException;
import com.squid.kraken.v4.api.core.EngineUtils;
import com.squid.kraken.v4.api.core.InvalidIdAPIException;
import com.squid.kraken.v4.api.core.JobServiceBaseImpl.OutputCompression;
import com.squid.kraken.v4.api.core.JobServiceBaseImpl.OutputFormat;
import com.squid.kraken.v4.api.core.JobStats;
import com.squid.kraken.v4.api.core.ObjectNotFoundAPIException;
import com.squid.kraken.v4.api.core.PerfDB;
import com.squid.kraken.v4.api.core.ServiceUtils;
import com.squid.kraken.v4.api.core.bookmark.BookmarkServiceBaseImpl;
import com.squid.kraken.v4.api.core.projectanalysisjob.AnalysisJobComputer;
import com.squid.kraken.v4.caching.NotInCacheException;
import com.squid.kraken.v4.caching.redis.RedisCacheManager;
import com.squid.kraken.v4.caching.redis.queryworkerserver.QueryWorkerJobStatus;
import com.squid.kraken.v4.core.analysis.datamatrix.DataMatrix;
import com.squid.kraken.v4.core.analysis.datamatrix.IDataMatrixConverter;
import com.squid.kraken.v4.core.analysis.datamatrix.RecordConverter;
import com.squid.kraken.v4.core.analysis.datamatrix.TransposeConverter;
import com.squid.kraken.v4.core.analysis.engine.bookmark.BookmarkManager;
import com.squid.kraken.v4.core.analysis.engine.hierarchy.DimensionIndex;
import com.squid.kraken.v4.core.analysis.engine.hierarchy.DimensionIndex.Status;
import com.squid.kraken.v4.core.analysis.engine.hierarchy.DomainHierarchy;
import com.squid.kraken.v4.core.analysis.engine.hierarchy.DomainHierarchyManager;
import com.squid.kraken.v4.core.analysis.engine.hierarchy.SegmentManager;
import com.squid.kraken.v4.core.analysis.engine.processor.ComputingException;
import com.squid.kraken.v4.core.analysis.engine.processor.ComputingService;
import com.squid.kraken.v4.core.analysis.engine.project.ProjectManager;
import com.squid.kraken.v4.core.analysis.model.DashboardAnalysis;
import com.squid.kraken.v4.core.analysis.model.DashboardSelection;
import com.squid.kraken.v4.core.analysis.scope.AxisExpression;
import com.squid.kraken.v4.core.analysis.scope.GlobalExpressionScope;
import com.squid.kraken.v4.core.analysis.scope.MeasureExpression;
import com.squid.kraken.v4.core.analysis.scope.SpaceExpression;
import com.squid.kraken.v4.core.analysis.scope.SpaceScope;
import com.squid.kraken.v4.core.analysis.scope.UniverseScope;
import com.squid.kraken.v4.core.analysis.universe.Axis;
import com.squid.kraken.v4.core.analysis.universe.Measure;
import com.squid.kraken.v4.core.analysis.universe.Space;
import com.squid.kraken.v4.core.analysis.universe.Universe;
import com.squid.kraken.v4.core.expression.reference.DomainReference;
import com.squid.kraken.v4.core.expression.scope.DomainExpressionScope;
import com.squid.kraken.v4.core.expression.scope.ExpressionSuggestionHandler;
import com.squid.kraken.v4.core.model.domain.DomainDomain;
import com.squid.kraken.v4.export.ExportSourceWriter;
import com.squid.kraken.v4.export.ExportSourceWriterCSV;
import com.squid.kraken.v4.export.ExportSourceWriterXLSX;
import com.squid.kraken.v4.model.AccessRight;
import com.squid.kraken.v4.model.AccessRight.Role;
import com.squid.kraken.v4.model.AnalyticsQuery;
import com.squid.kraken.v4.model.AnalyticsQueryImpl;
import com.squid.kraken.v4.model.AnalyticsReply;
import com.squid.kraken.v4.model.AnalyticsResult;
import com.squid.kraken.v4.model.AnalyticsResult.Info;
import com.squid.kraken.v4.model.Bookmark;
import com.squid.kraken.v4.model.BookmarkConfig;
import com.squid.kraken.v4.model.BookmarkFolderPK;
import com.squid.kraken.v4.model.BookmarkPK;
import com.squid.kraken.v4.model.DataTable;
import com.squid.kraken.v4.model.DataTable.Col;
import com.squid.kraken.v4.model.DataTable.Row;
import com.squid.kraken.v4.model.Dimension;
import com.squid.kraken.v4.model.Dimension.Type;
import com.squid.kraken.v4.model.Domain;
import com.squid.kraken.v4.model.Expression;
import com.squid.kraken.v4.model.ExpressionSuggestion;
import com.squid.kraken.v4.model.ExpressionSuggestionItem;
import com.squid.kraken.v4.model.Facet;
import com.squid.kraken.v4.model.FacetExpression;
import com.squid.kraken.v4.model.FacetMember;
import com.squid.kraken.v4.model.FacetMemberInterval;
import com.squid.kraken.v4.model.FacetMemberString;
import com.squid.kraken.v4.model.FacetSelection;
import com.squid.kraken.v4.model.Metric;
import com.squid.kraken.v4.model.NavigationItem;
import com.squid.kraken.v4.model.NavigationQuery;
import com.squid.kraken.v4.model.NavigationQuery.HierarchyMode;
import com.squid.kraken.v4.model.NavigationQuery.Style;
import com.squid.kraken.v4.model.NavigationQuery.Visibility;
import com.squid.kraken.v4.model.NavigationReply;
import com.squid.kraken.v4.model.NavigationResult;
import com.squid.kraken.v4.model.ObjectType;
import com.squid.kraken.v4.model.Problem;
import com.squid.kraken.v4.model.Problem.Severity;
import com.squid.kraken.v4.model.Project;
import com.squid.kraken.v4.model.ProjectAnalysisJob;
import com.squid.kraken.v4.model.ProjectAnalysisJob.Direction;
import com.squid.kraken.v4.model.ProjectAnalysisJob.Index;
import com.squid.kraken.v4.model.ProjectAnalysisJob.OrderBy;
import com.squid.kraken.v4.model.ProjectAnalysisJob.RollUp;
import com.squid.kraken.v4.model.ProjectAnalysisJobPK;
import com.squid.kraken.v4.model.ProjectFacetJob;
import com.squid.kraken.v4.model.ProjectPK;
import com.squid.kraken.v4.model.ValueType;
import com.squid.kraken.v4.model.ViewQuery;
import com.squid.kraken.v4.model.ViewReply;
import com.squid.kraken.v4.persistence.AppContext;
import com.squid.kraken.v4.persistence.DAOFactory;
import com.squid.kraken.v4.persistence.dao.ProjectDAO;
import com.squid.kraken.v4.vegalite.VegaliteConfigurator;
import com.squid.kraken.v4.vegalite.VegaliteSpecs;
import com.squid.kraken.v4.vegalite.VegaliteSpecs.Data;
import com.squid.kraken.v4.vegalite.VegaliteSpecs.DataType;
import com.squid.kraken.v4.vegalite.VegaliteSpecs.Encoding;
import com.squid.kraken.v4.vegalite.VegaliteSpecs.Format;
import com.squid.kraken.v4.vegalite.VegaliteSpecs.FormatType;
import com.squid.kraken.v4.vegalite.VegaliteSpecs.Mark;
import com.squid.kraken.v4.vegalite.VegaliteSpecs.Operation;
import com.squid.kraken.v4.vegalite.VegaliteSpecs.Order;
import com.squid.kraken.v4.vegalite.VegaliteSpecs.Sort;

/**
 * @author sergefantino
 *
 */
public class AnalyticsServiceBaseImpl implements AnalyticsServiceConstants {

    static final Logger logger = LoggerFactory.getLogger(AnalyticsServiceBaseImpl.class);

    private UriInfo uriInfo = null;

    private URI publicBaseUri = null;
    private AppContext userContext = null;

    protected AnalyticsServiceBaseImpl(UriInfo uriInfo, AppContext userContext) {
        this.uriInfo = uriInfo;
        this.publicBaseUri = getPublicBaseUri(uriInfo);
        this.userContext = userContext;
    }

    public UriBuilder getPublicBaseUriBuilder() {
        return new UriBuilderImpl(publicBaseUri);
    }

    /**
     * @param uriInfo2
     * @return
     */
    private URI getPublicBaseUri(UriInfo uriInfo) {
        // first check if there is a publicBaseUri parameter
        String uri = KrakenConfig.getProperty(KrakenConfig.publicBaseUri, true);
        if (uri != null) {
            try {
                return new URI(uri);
            } catch (URISyntaxException e) {
                // let's try the next
            }
        }
        // second, try to use the OAuth endpoint
        String oauthEndpoint = KrakenConfig.getProperty(KrakenConfig.krakenOAuthEndpoint, true);
        if (oauthEndpoint != null) {
            try {
                URI check = new URI(oauthEndpoint);
                // check that it is not using the ob.io central auth
                if (!"auth.openbouquet.io".equalsIgnoreCase(check.getHost())) {
                    while (oauthEndpoint.endsWith("/")) {
                        oauthEndpoint = oauthEndpoint.substring(0, oauthEndpoint.length() - 1);
                    }
                    if (oauthEndpoint.endsWith("/auth/oauth")) {
                        oauthEndpoint = oauthEndpoint.substring(0, oauthEndpoint.length() - "/auth/oauth".length());
                        return new URI(oauthEndpoint + "/v4.2");
                    }
                }
            } catch (URISyntaxException e) {
                // let's try the next
            }
        }
        // last, use the uriInfo
        return uriInfo.getBaseUri();
    }

    private static final NavigationItem ROOT_FOLDER = new NavigationItem("Root",
            "list all your available content, organize by Projects and Bookmarks", null, "/", "FOLDER");
    private static final NavigationItem PROJECTS_FOLDER = new NavigationItem("Projects", "list all your Projects",
            "/", "/PROJECTS", "FOLDER");
    private static final NavigationItem SHARED_FOLDER = new NavigationItem("Shared Bookmarks",
            "list all the bookmarks shared with you", "/", "/SHARED", "FOLDER");
    private static final NavigationItem MYBOOKMARKS_FOLDER = new NavigationItem("My Bookmarks",
            "list all your bookmarks", "/", "/MYBOOKMARKS", "FOLDER");

    public Response listContent(AppContext userContext, String parent, String search, HierarchyMode hierarchyMode,
            Visibility visibility, Style style, String envelope) throws ScopeException {
        if (parent != null && parent.endsWith("/")) {
            parent = parent.substring(0, parent.length() - 1);// remove trailing /
        }
        NavigationQuery query = new NavigationQuery();
        query.setParent(parent);
        query.setQ(search);
        query.setHiearchy(hierarchyMode);
        query.setStyle(style != null ? style : Style.HUMAN);
        query.setVisibility(visibility != null ? visibility : Visibility.ALL);
        if (envelope == null) {
            if (query.getStyle() == Style.HUMAN || query.getStyle() == Style.HTML) {
                envelope = "RESULT";
            } else {
                envelope = "ALL";
            }
        }
        //
        // tokenize the search string
        String[] filters = null;
        if (search != null)
            filters = search.toLowerCase().split(" ");
        //
        NavigationResult result = new NavigationResult();
        result.setChildren(new ArrayList<NavigationItem>());
        if (parent == null || parent.length() == 0) {
            result.setParent(createLinkableFolder(userContext, query, ROOT_FOLDER));
            // this is the root
            result.getChildren().add(createLinkableFolder(userContext, query, PROJECTS_FOLDER));
            if (hierarchyMode != null)
                listProjects(userContext, query, PROJECTS_FOLDER.getSelfRef(), filters, hierarchyMode,
                        result.getChildren());
            result.getChildren().add(createLinkableFolder(userContext, query, SHARED_FOLDER));
            if (hierarchyMode != null)
                listSharedBoomarks(userContext, query, SHARED_FOLDER.getSelfRef(), filters, hierarchyMode,
                        result.getChildren());
            result.getChildren().add(createLinkableFolder(userContext, query, MYBOOKMARKS_FOLDER));
            if (hierarchyMode != null)
                listMyBoomarks(userContext, query, MYBOOKMARKS_FOLDER.getSelfRef(), filters, hierarchyMode,
                        result.getChildren());
        } else {
            // need to list parent's content
            if (parent.startsWith(PROJECTS_FOLDER.getSelfRef())) {
                result.setParent(
                        listProjects(userContext, query, parent, filters, hierarchyMode, result.getChildren()));
            } else if (parent.startsWith(SHARED_FOLDER.getSelfRef())) {
                result.setParent(listSharedBoomarks(userContext, query, parent, filters, hierarchyMode,
                        result.getChildren()));
            } else if (parent.startsWith(MYBOOKMARKS_FOLDER.getSelfRef())) {
                result.setParent(
                        listMyBoomarks(userContext, query, parent, filters, hierarchyMode, result.getChildren()));
            } else {
                // invalid
                throw new ObjectNotFoundAPIException("invalid parent reference", true);
            }
        }
        // sort content
        sortNavigationContent(result.getChildren());
        // parent
        if (result.getParent() != null) {
            result.getParent().setLink(createLinkToFolder(userContext, query, result.getParent().getSelfRef()));// self link
            if (result.getParent().getParentRef() != null && !result.getParent().getParentRef().equals("")) {
                result.getParent()
                        .setUpLink(createLinkToFolder(userContext, query, result.getParent().getParentRef()));// self link
            }
        }
        // create results
        if (style == Style.HTML) {
            return createHTMLPageList(userContext, query, result);
        } else if (envelope.equalsIgnoreCase("ALL")) {
            return Response.ok(new NavigationReply(query, result)).build();
        } else {// RESULT
            return Response.ok(result).build();
        }
    }

    private NavigationItem createLinkableFolder(AppContext userContext, NavigationQuery query,
            NavigationItem folder) {
        if (query.getStyle() == Style.HUMAN || query.getStyle() == Style.HTML) {
            NavigationItem copy = new NavigationItem(folder);
            copy.setLink(createLinkToFolder(userContext, query, copy));
            return copy;
        } else {
            return folder;
        }
    }

    private static final List<String> topLevelOrder = Arrays.asList(new String[] { PROJECTS_FOLDER.getSelfRef(),
            SHARED_FOLDER.getSelfRef(), MYBOOKMARKS_FOLDER.getSelfRef() });
    private static final List<String> typeOrder = Arrays.asList(
            new String[] { NavigationItem.FOLDER_TYPE, NavigationItem.BOOKMARK_TYPE, NavigationItem.DOMAIN_TYPE });

    /**
     * @param content
     */
    private void sortNavigationContent(List<NavigationItem> content) {
        Collections.sort(content, new Comparator<NavigationItem>() {
            @Override
            public int compare(NavigationItem o1, NavigationItem o2) {
                if (o1.getParentRef() == ROOT_FOLDER.getSelfRef()
                        && o2.getParentRef() == ROOT_FOLDER.getSelfRef()) {
                    // special rule for top level
                    return Integer.compare(topLevelOrder.indexOf(o1.getSelfRef()),
                            topLevelOrder.indexOf(o2.getSelfRef()));
                } else if (o1.getParentRef().equals(o2.getParentRef())) {
                    if (o1.getType().equals(o2.getType())) {
                        // sort by name
                        return o1.getName().compareTo(o2.getName());
                    } else {
                        // sort by type
                        return Integer.compare(typeOrder.indexOf(o1.getType()), typeOrder.indexOf(o2.getType()));
                    }
                } else {
                    // sort by parent
                    return o1.getParentRef().compareTo(o2.getParentRef());
                }
            }
        });
    }

    /**
     * list the projects and their content (domains) depending on the parent path.
     * Note that this method does not support the flat mode because of computing constraints.
     * @param userContext
     * @param query 
     * @param parent
     * @param content
     * @throws ScopeException
     */
    private NavigationItem listProjects(AppContext userContext, NavigationQuery query, String parent,
            String[] filters, HierarchyMode hierarchyMode, List<NavigationItem> content) throws ScopeException {
        // list project related resources
        if (parent == null || parent.equals("") || parent.equals("/")
                || parent.equals(PROJECTS_FOLDER.getSelfRef())) {
            // return available project
            List<Project> projects = ((ProjectDAO) DAOFactory.getDAOFactory().getDAO(Project.class))
                    .findByCustomer(userContext, userContext.getCustomerPk());
            for (Project project : projects) {
                if (filters == null || filter(project, filters)) {
                    NavigationItem folder = new NavigationItem(query, project, parent);
                    if (query.getStyle() == Style.HUMAN || query.getStyle() == Style.HTML) {
                        folder.setLink(createLinkToFolder(userContext, query, folder));
                        folder.setObjectLink(createObjectLink(userContext, query, project));
                    }
                    HashMap<String, String> attrs = new HashMap<>();
                    attrs.put("jdbc", project.getDbUrl());
                    folder.setAttributes(attrs);
                    content.add(folder);
                }
            }
            return createLinkableFolder(userContext, query, PROJECTS_FOLDER);
        } else if (parent.startsWith(PROJECTS_FOLDER.getSelfRef())) {
            String projectRef = parent.substring(PROJECTS_FOLDER.getSelfRef().length() + 1);// remove /PROJECTS/ part
            Project project = findProject(userContext, projectRef);
            List<Domain> domains = ProjectManager.INSTANCE.getDomains(userContext, project.getId());
            Visibility visibility = query.getVisibility();
            for (Domain domain : domains) {
                if (visibility == Visibility.ALL || isVisible(userContext, domain, visibility)) {
                    String name = domain.getName();
                    if (filters == null || filter(name, filters)) {
                        NavigationItem item = new NavigationItem(query, project, domain, parent);
                        if (query.getStyle() == Style.HUMAN || query.getStyle() == Style.HTML) {
                            item.setLink(createLinkToAnalysis(userContext, query, item));
                            item.setObjectLink(createObjectLink(userContext, query, domain));
                            item.setViewLink(createLinkToView(userContext, query, item));
                        }
                        HashMap<String, String> attrs = new HashMap<>();
                        attrs.put("project", project.getName());
                        item.setAttributes(attrs);
                        content.add(item);
                    }
                }
            }
            return new NavigationItem(query, project, PROJECTS_FOLDER.getSelfRef());
        } else {
            // what's this parent anyway???
            return null;
        }
    }

    /**
     * @param userContext
     * @param domain
     * @param visibility
     * @return
     */
    private boolean isVisible(AppContext ctx, Domain domain, Visibility visibility) {
        boolean visible = ProjectManager.INSTANCE.isVisible(ctx, domain);
        if (!visible)
            return false;
        if (visibility == Visibility.ALL)
            return true;
        if (visibility == Visibility.VISIBLE) {
            return !domain.isDynamic();
        } else if (visibility == Visibility.HIDDEN) {
            return domain.isDynamic();
        } else {
            return false;
        }
    }

    /**
     * create a link to the NavigationItem
     * @param item
     * @return
     */
    private URI createLinkToFolder(AppContext userContext, NavigationQuery query, NavigationItem item) {
        return createNavigationQuery(userContext, query).build(item.getSelfRef());
    }

    private URI createLinkToFolder(AppContext userContext, NavigationQuery query, String selfRef) {
        return createNavigationQuery(userContext, query).build(selfRef != null ? selfRef : "");
    }

    private URI createLinkToAnalysis(AppContext userContext, NavigationQuery query, NavigationItem item) {
        UriBuilder builder = getPublicBaseUriBuilder().path("/analytics/{" + BBID_PARAM_NAME + "}/query");
        if (query.getStyle() != null)
            builder.queryParam(STYLE_PARAM, query.getStyle());
        builder.queryParam("access_token", userContext.getToken().getOid());
        return builder.build(item.getSelfRef());
    }

    private URI createLinkToView(AppContext userContext, NavigationQuery query, NavigationItem item) {
        UriBuilder builder = getPublicBaseUriBuilder().path("/analytics/{" + BBID_PARAM_NAME + "}/view");
        if (query.getStyle() != null)
            builder.queryParam(STYLE_PARAM, query.getStyle());
        builder.queryParam("access_token", userContext.getToken().getOid());
        return builder.build(item.getSelfRef());
    }

    private UriBuilder createNavigationQuery(AppContext userContext, NavigationQuery query) {
        UriBuilder builder = getPublicBaseUriBuilder().path("/analytics").queryParam("parent", "{PARENT}");
        if (query.getHiearchy() != null)
            builder.queryParam("hierarchy", query.getHiearchy());
        if (query.getStyle() != null)
            builder.queryParam(STYLE_PARAM, query.getStyle());
        if (query.getVisibility() != null)
            builder.queryParam(VISIBILITY_PARAM, query.getVisibility());
        builder.queryParam("access_token", userContext.getToken().getOid());
        return builder;
    }

    private URI createObjectLink(AppContext userContext, NavigationQuery query, Project project) {
        UriBuilder builder = getPublicBaseUriBuilder().path("/rs/projects/{projectID}");
        if (query.getStyle() != null)
            builder.queryParam(STYLE_PARAM, query.getStyle());
        builder.queryParam("access_token", userContext.getToken().getOid());
        return builder.build(project.getOid());
    }

    private URI createObjectLink(AppContext userContext, NavigationQuery query, Domain domain) {
        UriBuilder builder = getPublicBaseUriBuilder().path("/rs/projects/{projectID}/domains/{domainID}");
        if (query.getStyle() != null)
            builder.queryParam(STYLE_PARAM, query.getStyle());
        builder.queryParam("access_token", userContext.getToken().getOid());
        return builder.build(domain.getId().getProjectId(), domain.getOid());
    }

    private URI createObjectLink(AppContext userContext, NavigationQuery query, Bookmark bookmark) {
        UriBuilder builder = getPublicBaseUriBuilder().path("/rs/projects/{projectID}/bookmarks/{domainID}");
        if (query.getStyle() != null)
            builder.queryParam(STYLE_PARAM, query.getStyle());
        builder.queryParam("access_token", userContext.getToken().getOid());
        return builder.build(bookmark.getId().getProjectId(), bookmark.getOid());
    }

    /**
     * @param userContext
     * @param projectRef
     * @return
     * @throws ScopeException 
     */
    private Project findProject(AppContext userContext, String projectRef) throws ScopeException {
        if (projectRef.startsWith("@")) {
            // using ID
            String projectId = projectRef.substring(1);
            ProjectPK projectPk = new ProjectPK(userContext.getClientId(), projectId);
            return ProjectManager.INSTANCE.getProject(userContext, projectPk);
        } else {
            // using name
            List<Project> projects = ((ProjectDAO) DAOFactory.getDAOFactory().getDAO(Project.class))
                    .findByCustomer(userContext, userContext.getCustomerPk());
            for (Project project : projects) {
                if (project.getName() != null && project.getName().equals(projectRef)) {
                    return project;
                }
            }
            throw new ScopeException("cannot find project with name='" + projectRef + "'");
        }
    }

    /**
     * list the MyBookmarks content (bookmarks and folders)
     * @param userContext
     * @param query 
     * @param parent
     * @param isFlat
     * @param content
     * @return the parent folder
     * @throws ScopeException
     */
    private NavigationItem listMyBoomarks(AppContext userContext, NavigationQuery query, String parent,
            String[] filters, HierarchyMode hierarchyMode, List<NavigationItem> content) throws ScopeException {
        // list mybookmark related resources
        String fullPath = BookmarkManager.INSTANCE.getMyBookmarkPath(userContext);
        NavigationItem parentFolder = null;
        if (parent.equals(MYBOOKMARKS_FOLDER.getSelfRef())) {
            // just keep the fullpath
            parentFolder = createLinkableFolder(userContext, query, MYBOOKMARKS_FOLDER);
        } else {
            // add the remaining path to fullpath
            fullPath += parent.substring(MYBOOKMARKS_FOLDER.getSelfRef().length());
            String name = parent.substring(parent.lastIndexOf("/"));
            String grandParent = parent.substring(0, parent.lastIndexOf("/"));
            parentFolder = new NavigationItem(name, "", grandParent, parent, NavigationItem.FOLDER_TYPE);
        }
        listBoomarks(userContext, query, parent, filters, hierarchyMode, fullPath, content);
        return parentFolder;
    }

    /**
     * List the Shared bookmarks and folders
     * @param userContext
     * @param query 
     * @param parent
     * @param isFlat
     * @param content
     * @return the parent folder
     * @throws ScopeException
     */
    private NavigationItem listSharedBoomarks(AppContext userContext, NavigationQuery query, String parent,
            String[] filters, HierarchyMode hierarchyMode, List<NavigationItem> content) throws ScopeException {
        // list mybookmark related resources
        String fullPath = Bookmark.SEPARATOR + Bookmark.Folder.SHARED;
        NavigationItem parentFolder = null;
        if (parent.equals(SHARED_FOLDER.getSelfRef())) {
            parentFolder = createLinkableFolder(userContext, query, SHARED_FOLDER);
        } else {
            // add the remaining path to fullpath
            fullPath += parent.substring(SHARED_FOLDER.getSelfRef().length());
            String name = parent.substring(parent.lastIndexOf("/"));
            String grandParent = parent.substring(0, parent.lastIndexOf("/"));
            parentFolder = new NavigationItem(name, "", grandParent, parent, NavigationItem.FOLDER_TYPE);
        }
        listBoomarks(userContext, query, parent, filters, hierarchyMode, fullPath, content);
        return parentFolder;
    }

    /**
     * 
     * @param userContext
     * @param query
     * @param parent
     * @param filters
     * @param hierarchyMode
     * @param fullPath
     * @param content
     * @throws ScopeException
     */
    private void listBoomarks(AppContext userContext, NavigationQuery query, String parent, String[] filters,
            HierarchyMode hierarchyMode, String fullPath, List<NavigationItem> content) throws ScopeException {
        // list the content first
        List<Bookmark> bookmarks = BookmarkManager.INSTANCE.findBookmarksByParent(userContext, fullPath);
        HashSet<String> folders = new HashSet<>();
        for (Bookmark bookmark : bookmarks) {
            Project project = ProjectManager.INSTANCE.getProject(userContext, bookmark.getId().getParent());
            String path = bookmark.getPath();
            // only handle the exact path
            boolean checkParent = (hierarchyMode == HierarchyMode.FLAT) ? path.startsWith(fullPath)
                    : path.equals(fullPath);
            if (checkParent) {
                if (filters == null || filter(bookmark, project, filters)) {
                    String actualParent = parent;
                    if (hierarchyMode == HierarchyMode.FLAT) {
                        actualParent = (parent + path.substring(fullPath.length()));
                    }
                    NavigationItem item = new NavigationItem(query, project, bookmark, actualParent);
                    if (query.getStyle() == Style.HUMAN || query.getStyle() == Style.HTML) {
                        item.setLink(createLinkToAnalysis(userContext, query, item));
                        item.setViewLink(createLinkToView(userContext, query, item));
                        item.setObjectLink(createObjectLink(userContext, query, bookmark));
                    }
                    HashMap<String, String> attrs = new HashMap<>();
                    attrs.put("project", project.getName());
                    item.setAttributes(attrs);
                    content.add(item);
                }
            } else {
                // it's a sub folder
                path = bookmark.getPath().substring(fullPath.length() + 1);// remove first /
                String[] split = path.split("/");
                if (split.length > 0) {
                    String name = "/" + (hierarchyMode != null ? path : split[0]);
                    String selfpath = parent + name;
                    if (!folders.contains(selfpath)) {
                        if (filters == null || filter(name, filters)) {
                            String oid = Base64.encodeBase64URLSafeString(selfpath.getBytes());
                            // legacy folder PK support
                            BookmarkFolderPK id = new BookmarkFolderPK(bookmark.getId().getCustomerId(), oid);
                            NavigationItem folder = new NavigationItem(query.getStyle() == Style.LEGACY ? id : null,
                                    name, "", parent, selfpath, NavigationItem.FOLDER_TYPE);
                            if (query.getStyle() == Style.HUMAN || query.getStyle() == Style.HTML) {
                                folder.setLink(createLinkToFolder(userContext, query, folder));
                            }
                            content.add(folder);
                            folders.add(selfpath);
                        }
                    }
                }
            }
        }
    }

    private boolean filter(Project project, String[] filters) {
        final String name = project.getName();
        final String description = project.getDescription();
        final String attr = project.getDbUrl();
        for (String filter : filters) {
            if (name == null || !name.toLowerCase().contains(filter)) {
                if (description == null || !description.toLowerCase().contains(filter)) {
                    if (attr == null || !attr.toLowerCase().contains(filter)) {
                        return false;
                    }
                }
            }
        }
        return true;
    }

    private boolean filter(Bookmark bookmark, Project project, String[] filters) {
        final String name = bookmark.getName();
        final String description = bookmark.getDescription();
        final String attr = project.getName();
        for (String filter : filters) {
            if (name == null || !name.toLowerCase().contains(filter)) {
                if (description == null || !description.toLowerCase().contains(filter)) {
                    if (attr == null || !attr.toLowerCase().contains(filter)) {
                        return false;
                    }
                }
            }
        }
        return true;
    }

    private boolean filter(String name, String[] filters) {
        if (name == null)
            return false;
        for (String filter : filters) {
            if (!name.toLowerCase().contains(filter)) {
                return false;
            }
        }
        return true;
    }

    /**
     * @param userContext
     * @param query
     * @param parent
     * @param parent2 
     * @return
     */
    public Bookmark createBookmark(AppContext userContext, AnalyticsQuery query, String BBID, String name,
            String parent) {
        try {
            Space space = getSpace(userContext, BBID);
            if (query == null) {
                throw new APIException("undefined query");
            }
            if (query.getDomain() == null || query.getDomain() == "") {
                throw new APIException("undefined domain");
            }
            String domainID = "@'" + space.getDomain().getOid() + "'";
            if (!query.getDomain().equals(domainID)) {
                throw new APIException("invalid domain definition for the query, doesn't not match the REFERENCE");
            }
            Bookmark bookmark = new Bookmark();
            BookmarkPK bookmarkPK = new BookmarkPK(space.getUniverse().getProject().getId());
            bookmark.setId(bookmarkPK);
            BookmarkConfig config = createBookmarkConfig(space, query);
            ObjectMapper mapper = new ObjectMapper();
            mapper.configure(DeserializationFeature.FAIL_ON_UNKNOWN_PROPERTIES, false);
            String json = mapper.writeValueAsString(config);
            JsonNode tree = mapper.readTree(json);
            bookmark.setConfig(tree);
            bookmark.setDescription("created using the super cool new Bookmark API");
            bookmark.setName(name);
            String path = "";
            if (parent.startsWith(MYBOOKMARKS_FOLDER.getSelfRef())) {
                path = parent.substring(MYBOOKMARKS_FOLDER.getSelfRef().length());
                path = BookmarkManager.INSTANCE.getMyBookmarkPath(userContext) + path;
            } else if (parent.startsWith(SHARED_FOLDER.getSelfRef())) {
                path = parent.substring(SHARED_FOLDER.getSelfRef().length());
                path += Bookmark.SEPARATOR + Bookmark.Folder.SHARED + path;
            } else if (!parent.startsWith("/")) {
                path = BookmarkManager.INSTANCE.getMyBookmarkPath(userContext) + "/" + parent;
            } else {
                throw new ObjectNotFoundAPIException("unable to save a bookmark in this path: " + parent, true);
            }
            bookmark.setPath(path);
            //
            BookmarkServiceBaseImpl.getInstance().store(userContext, bookmark);
            return bookmark;
        } catch (IOException e) {
            throw new APIException("cannot create the bookmark: JSON error: " + e.getMessage());
        } catch (ScopeException e) {
            throw new ObjectNotFoundAPIException("invalid REFERENCE :" + e.getMessage(), true);
        }
    }

    public Object getItem(AppContext userContext, String BBID) throws ScopeException {
        Space space = getSpace(userContext, BBID);
        if (space.getParent() != null) {
            throw new ScopeException("invalid ID");
        }
        if (space.hasBookmark()) {
            return space.getBookmark();
        } else {
            return space.getDomain();
        }
    }

    private Space getSpace(AppContext userContext, String BBID) {
        try {
            GlobalExpressionScope scope = new GlobalExpressionScope(userContext);
            ExpressionAST expr = scope.parseExpression(BBID);
            if (expr instanceof SpaceExpression) {
                SpaceExpression ref = (SpaceExpression) expr;
                Space space = ref.getSpace();
                return space;
            }
        } catch (ScopeException e) {
            throw new ObjectNotFoundAPIException("invalid REFERENCE: " + e.getMessage(), true);
        }
        // else
        throw new ObjectNotFoundAPIException("invalid REFERENCE", true);
    }

    public Facet getFacet(AppContext userContext, String BBID, String facetId, String search, String[] filters,
            Integer maxResults, Integer startIndex, Integer timeoutMs) throws ComputingException {
        try {
            Space space = getSpace(userContext, BBID);
            Domain domain = space.getDomain();
            //
            AnalyticsQuery query = new AnalyticsQueryImpl();
            if ((filters != null) && (filters.length > 0)) {
                query.setFilters(Arrays.asList(filters));
            }
            //
            ProjectFacetJob job = new ProjectFacetJob();
            job.setDomain(Collections.singletonList(domain.getId()));
            job.setCustomerId(userContext.getCustomerId());
            //
            BookmarkConfig config = null;
            if (space.hasBookmark()) {
                config = BookmarkManager.INSTANCE.readConfig(space.getBookmark());
            }
            // merge the config with the query
            mergeBoomarkConfig(space, query, config);
            // extract the facet selection
            FacetSelection selection = createFacetSelection(space, query);
            job.setSelection(selection);
            //
            Universe universe = space.getUniverse();
            List<Domain> domains = Collections.singletonList(domain);
            DashboardSelection sel = EngineUtils.getInstance().applyFacetSelection(userContext, universe, domains,
                    job.getSelection());
            //return ComputingService.INSTANCE.glitterFacet(universe, domain, ds, axis, filter, startIndex, maxResults, timeoutMs);
            if (SegmentManager.isSegmentFacet(facetId)) {
                DomainHierarchy hierarchy = universe.getDomainHierarchy(domain, true);
                return SegmentManager.createSegmentFacet(universe, hierarchy, domain, facetId, search, maxResults,
                        startIndex, sel);
            } else {
                /*
                Axis axis = EngineUtils.getInstance().getFacetAxis(userContext,
                      universe, facetId);// universe.axis(facetId);
                Domain domain2 = axis.getParent().getTop().getDomain();
                //
                if (!domain2.equals(domain)) {
                   DimensionIndex index = axis.getIndex();
                   if (index!=null) {
                      throw new ScopeException("cannot list the facet for '"+index.getDimensionName()+"': not in the job scope");
                   } else {
                      throw new ScopeException("cannot list the facet for '"+axis.prettyPrint()+"': not in the job scope");
                   }
                }
                */
                // using the bookmark scope instead
                SpaceScope scope = new SpaceScope(universe.S(domain));
                ExpressionAST expr = scope.parseExpression(facetId);
                Axis axis = universe.asAxis(expr);
                if (axis == null) {
                    throw new ScopeException("the expression '" + facetId
                            + "' doesn't resolve to an valid facet in this bookmark scope");
                }
                //
                Facet facet = ComputingService.INSTANCE.glitterFacet(universe, domain, sel, axis, search,
                        startIndex != null ? startIndex : 0, maxResults != null ? maxResults : 50, timeoutMs);

                if (facet == null) {
                    throw new ObjectNotFoundAPIException("no facet found with id : " + facetId,
                            userContext.isNoError());
                }
                return facet;
                // KRKN-53: if cannot compute the facet, just return with error informations
                /*
                if (facet.isError()) {
                   throw new APIException(facet.getErrorMessage(),
                 userContext.isNoError(), ApiError.COMPUTING_FAILED);
                }
                */
            }
        } catch (ScopeException | ComputingException | InterruptedException e) {
            throw new ComputingException(e.getLocalizedMessage(), e);
        } catch (TimeoutException e) {
            throw new ComputingInProgressAPIException(null, userContext.isNoError(), null);
        }
    }

    public Response scopeAnalysis(AppContext userContext, String BBID, String value, Integer offset,
            ObjectType[] types, ValueType[] values, Style style) throws ScopeException {
        if (value == null)
            value = "";
        Space space = getSpace(userContext, BBID);
        //
        DomainExpressionScope scope = new DomainExpressionScope(space.getUniverse(), space.getDomain());
        //SpaceScope scope = new SpaceScope(space);
        //
        ExpressionSuggestionHandler handler = new ExpressionSuggestionHandler(scope);
        if (offset == null) {
            offset = value.length() + 1;
        }
        Collection<ObjectType> typeFilters = null;
        if (types != null) {
            if (types.length > 1) {
                typeFilters = new HashSet<>(Arrays.asList(types));
            } else if (types.length == 1) {
                typeFilters = Collections.singletonList(types[0]);
            }
        }
        Collection<ValueType> valueFilters = null;
        if (values != null) {
            if (values.length > 1) {
                valueFilters = new HashSet<>(Arrays.asList(values));
            } else if (values.length == 1) {
                valueFilters = Collections.singletonList(values[0]);
            }
        }
        ExpressionSuggestion suggestions = handler.getSuggestion(value, offset, typeFilters, valueFilters);
        if (style == Style.HTML) {
            return createHTMLPageScope(space, suggestions, BBID, value, types, values);
        } else {
            return Response.ok(suggestions).build();
        }
    }

    public Response runAnalysis(final AppContext userContext, String BBID, final AnalyticsQuery query, String data,
            String envelope, Integer timeout) {
        Space space = null;// if we can initialize it, fine to report in the catch block
        try {
            //
            if (envelope == null) {
                envelope = computeEnvelope(query);
            }
            //
            space = getSpace(userContext, BBID);
            //
            Bookmark bookmark = space.getBookmark();
            BookmarkConfig config = BookmarkManager.INSTANCE.readConfig(bookmark);
            //
            // merge the bookmark config with the query
            mergeBoomarkConfig(space, query, config);
            if (query.getLimit() == null) {
                query.setLimit((long) 100);
            }
            // create the facet selection
            FacetSelection selection = createFacetSelection(space, query);
            final ProjectAnalysisJob job = createAnalysisJob(space.getUniverse(), query, selection,
                    OutputFormat.JSON);
            //
            final boolean lazyFlag = (query.getLazy() != null)
                    && (query.getLazy().equals("true") || query.getLazy().equals("noError"));
            //
            // create the AnalysisResult
            AnalyticsReply reply = new AnalyticsReply();
            if (query.getStyle() == Style.LEGACY) {
                // legacy may use the facetSelection
                reply.setSelection(selection);
            }
            reply.setQuery(query);
            //
            if (data == null || data.equals(""))
                data = "LEGACY";
            if (data.equalsIgnoreCase("SQL")) {
                // bypassing the ComputingService
                AnalysisJobComputer computer = new AnalysisJobComputer();
                String sql = computer.viewSQL(userContext, job);
                reply.setResult(sql);
            } else {
                if (query.getStyle() == Style.HTML) {
                    // change data format to legacy
                    data = "LEGACY";
                    if (query.getLimit() > 100 && query.getMaxResults() == null) {
                        // try to apply maxResults
                        query.setMaxResults(100);
                    }
                }
                try {
                    Callable<DataMatrix> task = new Callable<DataMatrix>() {
                        @Override
                        public DataMatrix call() throws Exception {
                            return compute(userContext, job, query.getMaxResults(), query.getStartIndex(),
                                    lazyFlag);
                        }
                    };
                    Future<DataMatrix> futur = ExecutionManager.INSTANCE.submit(userContext.getCustomerId(), task);
                    DataMatrix matrix = null;
                    if (timeout == null) {
                        // using the customer execution engine to control load
                        matrix = futur.get();
                    } else {
                        matrix = futur.get(timeout > 1000 ? timeout : 1000,
                                java.util.concurrent.TimeUnit.MILLISECONDS);
                    }
                    if (data == null || data.equals("") || data.equalsIgnoreCase("LEGACY")) {
                        DataTable legacy = matrix.toDataTable(userContext, query.getMaxResults(),
                                query.getStartIndex(), false, null);
                        reply.setResult(legacy);
                    } else {
                        IDataMatrixConverter<Object[]> converter = getConverter(data);
                        AnalyticsResult result = new AnalyticsResult();
                        Object[] output = converter.convert(query, matrix);
                        result.setData(output);
                        result.setInfo(getAnalyticsResultInfo(output.length, query.getStartIndex(), matrix));
                        reply.setResult(result);
                    }
                } catch (NotInCacheException e) {
                    if (query.getLazy().equals("noError")) {
                        reply.setResult(new AnalyticsResult());
                    } else {
                        throw e;
                    }
                } catch (ExecutionException e) {
                    if (query.getStyle() == Style.HUMAN || query.getStyle() == Style.HTML) {
                        // wrap the exception in a Problem
                        Throwable cause = getCauseException(e);
                        query.add(new Problem(Severity.ERROR, "SQL",
                                "Failed to run the query: " + cause.getMessage(), cause));
                    } else {
                        // just let if go
                        throwCauseException(e);
                    }
                } catch (TimeoutException e) {
                    if (query.getStyle() == Style.HUMAN || query.getStyle() == Style.HTML) {
                        URI link = getPublicBaseUriBuilder().path("/status/{queryID}")
                                .queryParam("access_token", userContext.getToken().getOid())
                                .build(query.getQueryID());
                        throw new ComputingInProgressAPIException("computing in progress", true, timeout * 2,
                                query.getQueryID(), link);
                    } else {
                        throw new ComputingInProgressAPIException("computing in progress", true, timeout * 2,
                                query.getQueryID());
                    }
                }
            }
            if (query.getStyle() == Style.HTML && data.equalsIgnoreCase("SQL")) {
                return createHTMLsql(reply.getResult().toString());
            } else if (query.getStyle() == Style.HTML && data.equalsIgnoreCase("LEGACY")) {
                return createHTMLPageTable(userContext, space, query, (DataTable) reply.getResult());
            } else if (envelope.equalsIgnoreCase("ALL")) {
                return Response.ok(reply).build();
            } else if (envelope.equalsIgnoreCase("RESULT")) {
                return Response.ok(reply.getResult()).build();
            } else if (envelope.equalsIgnoreCase("DATA")) {
                if (reply.getResult() instanceof AnalyticsResult) {
                    return Response.ok(((AnalyticsResult) reply.getResult()).getData()).build();
                } else if (reply.getResult() instanceof DataTable) {
                    return Response.ok(((DataTable) reply.getResult()).getRows()).build();
                } else {
                    // return result instead
                    return Response.ok(reply.getResult()).build();
                }
            }
            //else
            return Response.ok(reply).build();
        } catch (DatabaseServiceException | ComputingException | InterruptedException | ScopeException
                | SQLScopeException | RenderingException e) {
            if (query.getStyle() == Style.HTML) {
                query.add(new Problem(Severity.ERROR, "query",
                        "unable to run the query, fatal error: " + e.getMessage(), e));
                return createHTMLPageTable(userContext, space, query, null);
            } else {
                throw new APIException(e.getMessage(), true);
            }
        }
    }

    /**
     * Try to find the most relevant exception in the stack
     * @param the execution exception
     */
    private void throwCauseException(ExecutionException e) {
        Throwable cause = getCauseException(e);
        throwAPIException(cause);
    }

    private Throwable getCauseException(ExecutionException e) {
        Throwable previous = e;
        Throwable last = e;
        while (last.getCause() != null) {
            previous = last;
            last = last.getCause();
        }
        if (previous.getMessage() != null) {
            return previous;
        } else {
            return last;
        }
    }

    private String computeEnvelope(AnalyticsQuery query) {
        if (query.getStyle() == Style.HUMAN || query.getStyle() == Style.HTML) {
            return "ALL";
        } else if (query.getStyle() == Style.LEGACY) {
            return "RESULT";
        } else {//MACHINE
            return "ALL";
        }
    }

    /**
     * @param previous
     * @return
     */
    private void throwAPIException(Throwable exception) throws APIException {
        if (exception instanceof APIException) {
            throw (APIException) exception;
        } else {
            throw new APIException(exception, true);
        }
    }

    /**
     * @param userContext 
     * @param dataTable 
     * @return
     */
    private Response createHTMLPageTable(AppContext userContext, Space space, AnalyticsQuery query,
            DataTable data) {
        String title = space != null ? getPageTitle(space) : null;
        StringBuilder html = createHTMLHeader("Query: " + title);
        createHTMLtitle(html, title, query.getBBID(), getParentLink(space));
        createHTMLproblems(html, query.getProblems());
        if (data != null) {
            html.append("<table class='data'><tr>");
            html.append("<th></th>");
            for (Col col : data.getCols()) {
                html.append("<th>" + col.getName() + "</th>");
            }
            html.append("</tr>");
            int i = 1;
            if (query.getStartIndex() != null) {
                i = query.getStartIndex() + 1;
            }
            for (Row row : data.getRows()) {
                html.append("<tr>");
                html.append("<td valign='top'>#" + (i++) + "</td>");
                for (Col col : data.getCols()) {
                    Object value = row.getV()[col.getPos()];
                    if (value != null) {
                        if (col.getFormat() != null && col.getFormat().length() > 0) {
                            try {
                                value = String.format(col.getFormat(), value);
                            } catch (IllegalFormatException e) {
                                // ignore
                            }
                        }
                    } else {
                        value = "";
                    }
                    html.append("<td valign='top'>" + value + "</td>");
                }
                html.append("</tr>");
            }
            html.append("</table>");
            createHTMLpagination(html, query, data);
        } else {
            html.append("<i>Result is not available, it's probably due to an error</i>");
            html.append("<p>");
            createHTMLdataLinks(html, query);
            html.append("</p<br>");
        }
        html.append("<form>");
        createHTMLfilters(html, query);
        {
            UriBuilder builder = getPublicBaseUriBuilder().path("/analytics/{" + BBID_PARAM_NAME + "}/query");
            builder.queryParam(STYLE_PARAM, Style.HTML);
            builder.queryParam("access_token", userContext.getToken().getOid());
            URI uri = builder.build(space.getBBID(Style.ROBOT));
            html.append("<a href=\"" + StringEscapeUtils.escapeHtml4(uri.toString()) + "\">"
                    + StringEscapeUtils.escapeHtml4(space.getBBID(Style.HUMAN)) + "</a>");
        }
        html.append("<table>");
        html.append("<tr><td valign='top'>groupBy</td><td>");
        createHTMLinputArray(html, "text", "groupBy", query.getGroupBy());
        html.append(
                "</td><td valign='top'><p><i>Define the group-by facets to apply to results. Facet can be defined using it's ID or any valid expression. If empty, the subject default parameters will apply. You can use the * token to extend the subject default parameters.</i></p>");
        html.append("</td></tr>");
        html.append("<tr><td valign='top'>metrics</td><td>");
        createHTMLinputArray(html, "text", "metrics", query.getMetrics());
        html.append(
                "</td><td valign='top'><p><i>Define the metrics to compute. Metric can be defined using it's ID or any valid expression. If empty, the subject default parameters will apply. You can use the * token to extend the subject default parameters.</i></p>");
        html.append("</td></tr>");
        html.append("<tr><td valign='top'>orderBy</td><td>");
        createHTMLinputArray(html, "text", "orderBy", query.getOrderBy());
        html.append("</td></tr>");
        html.append("<tr><td>limit</td><td>");
        html.append("<input type=\"text\" name=\"limit\" value=\"" + getFieldValue(query.getLimit(), 0) + "\">");
        html.append("</td></tr>");
        html.append("<tr><td>maxResults</td><td>");
        html.append("<input type=\"text\" name=\"maxResults\" value=\"" + getFieldValue(query.getMaxResults(), 100)
                + "\">");
        html.append("</td></tr>");
        html.append("<tr><td>startIndex</td><td>");
        html.append("<input type=\"text\" name=\"startIndex\" value=\"" + getFieldValue(query.getStartIndex(), 0)
                + "\"></td><td><i>index is zero-based, so use the #count of the last row to view the next page</i>");
        html.append("</td></tr>");
        html.append("</table>" + "<input type=\"hidden\" name=\"style\" value=\"HTML\">"
                + "<input type=\"hidden\" name=\"access_token\" value=\"" + userContext.getToken().getOid() + "\">"
                + "<input type=\"submit\" value=\"Refresh\">" + "</form>");
        if (space != null)
            createHTMLscope(html, space, query);
        createHTMLAPIpanel(html, "runAnalysis");
        html.append("</body></html>");
        return Response.ok(html.toString(), "text/html").build();
    }

    private void createHTMLinputArray(StringBuilder html, String type, String name, List<? extends Object> values) {
        if (values == null || values.isEmpty()) {
            html.append("<input type=\"" + type + "\" size=100 name=\"" + name
                    + "\" value=\"\" placeholder=\"type formula\">");
        } else {
            boolean first = true;
            for (Object value : values) {
                if (!first)
                    html.append("<br>");
                else
                    first = false;
                html.append("<input type=\"" + type + "\" size=100 name=\"" + name + "\" value=\""
                        + getFieldValue(value.toString()) + "\">");
            }
            if (!first)
                html.append("<br>");
            html.append("<input type=\"" + type + "\" size=100 name=\"" + name
                    + "\" value=\"\" placeholder=\"type formula\">");
        }
    }

    private void createHTMLpagination(StringBuilder html, AnalyticsQuery query, DataTable data) {
        long lastRow = (data.getStartIndex() + data.getRows().size());
        long firstRow = data.getRows().size() > 0 ? (data.getStartIndex() + 1) : 0;
        html.append("<br><div>rows from " + firstRow + " to " + lastRow + " out of " + data.getTotalSize()
                + " records");
        if (data.getFullset()) {
            html.append(" (the query is complete)");
        } else {
            html.append(" (the query has more data)");
        }
        if (lastRow < data.getTotalSize()) {
            // go to next page
            HashMap<String, Object> override = new HashMap<>();
            override.put(START_INDEX_PARAM, lastRow);
            URI nextLink = buildAnalyticsQueryURI(userContext, query, null, null, Style.HTML, override);
            html.append("&nbsp;[<a href=\"" + StringEscapeUtils.escapeHtml4(nextLink.toString()) + "\">next</a>]");
        }
        html.append("</div><div>");
        if (data.isFromSmartCache()) {
            html.append("data from smart-cache, last computed " + data.getExecutionDate());
        } else if (data.isFromCache()) {
            html.append("data from cache, last computed " + data.getExecutionDate());
        } else {
            html.append("fresh data just computed at " + data.getExecutionDate());
        }
        // add links
        createHTMLdataLinks(html, query);
        html.append("</div><br>");
    }

    private void createHTMLdataLinks(StringBuilder html, AnalyticsQuery query) {
        // add links
        { // for View
            HashMap<String, Object> override = new HashMap<>();
            override.put(LIMIT_PARAM, null);
            override.put(MAX_RESULTS_PARAM, null);
            URI sqlLink = buildAnalyticsViewURI(userContext, new ViewQuery(query), null, "ALL", Style.HTML,
                    override);//(userContext, query, "SQL", null, Style.HTML, null);
            html.append("&nbsp;[<a href=\"" + StringEscapeUtils.escapeHtml4(sqlLink.toString()) + "\">View</a>]");
        }
        { // for SQL
            URI sqlLink = buildAnalyticsQueryURI(userContext, query, "SQL", null, Style.HTML, null);
            html.append("&nbsp;[<a href=\"" + StringEscapeUtils.escapeHtml4(sqlLink.toString()) + "\">SQL</a>]");
        }
        { // for CSV export
            URI csvExport = buildAnalyticsExportURI(userContext, query, ".csv");
            html.append("&nbsp;[<a href=\"" + StringEscapeUtils.escapeHtml4(csvExport.toString())
                    + "\">Export CSV</a>]");
        }
        { // for XLS export
            URI xlsExport = buildAnalyticsExportURI(userContext, query, ".xls");
            html.append("&nbsp;[<a href=\"" + StringEscapeUtils.escapeHtml4(xlsExport.toString())
                    + "\">Export XLS</a>]");
        }
    }

    private void createHTMLpagination(StringBuilder html, ViewQuery query, Info info) {
        long lastRow = (info.getStartIndex() + info.getPageSize());
        long firstRow = info.getTotalSize() > 0 ? (info.getStartIndex() + 1) : 0;
        html.append("<br><div>rows from " + firstRow + " to " + lastRow + " out of " + info.getTotalSize()
                + " records");
        if (info.isComplete()) {
            html.append(" (the query is complete)");
        } else {
            html.append(" (the query has more data)");
        }
        if (lastRow < info.getTotalSize()) {
            // go to next page
            HashMap<String, Object> override = new HashMap<>();
            override.put(START_INDEX_PARAM, lastRow);
            URI nextLink = buildAnalyticsViewURI(userContext, query, null, null, Style.HTML, override);
            html.append("&nbsp;[<a href=\"" + StringEscapeUtils.escapeHtml4(nextLink.toString()) + "\">next</a>]");
        }
        html.append("</p>");
        if (info.isFromSmartCache()) {
            html.append("<p>data from smart-cache, last computed " + info.getExecutionDate() + "</p>");
        } else if (info.isFromCache()) {
            html.append("<p>data from cache, last computed " + info.getExecutionDate() + "</p>");
        } else {
            html.append("<p>fresh data just computed at " + info.getExecutionDate() + "</p>");
        }
    }

    /**
     * @param string
     * @return
     */
    private Response createHTMLsql(String sql) {
        StringBuilder html = new StringBuilder("<html><head>");
        html.append(
                "<script src='https://cdn.rawgit.com/google/code-prettify/master/loader/run_prettify.js?lang=sql'></script>");
        html.append("</head><body>");
        html.append(
                "<pre class='prettyprint lang-sql' style='white-space: pre-wrap;white-space: -moz-pre-wrap;white-space: -pre-wrap;white-space: -o-pre-wrap;word-wrap: break-word;padding:0px;margin:0px'>");
        html.append(StringEscapeUtils.escapeHtml4(sql));
        html.append("</pre>");
        html.append("</body></html>");
        return Response.ok(html.toString(), "text/html").build();
    }

    public Response exportAnalysis(final AppContext userContext, String BBID, final AnalyticsQuery query,
            final String filename, String fileext, String compression) {
        try {
            Space space = getSpace(userContext, BBID);
            //
            Bookmark bookmark = space.getBookmark();
            BookmarkConfig config = BookmarkManager.INSTANCE.readConfig(bookmark);
            //
            // merge the bookmark config with the query
            mergeBoomarkConfig(space, query, config);
            // create the facet selection
            FacetSelection selection = createFacetSelection(space, query);
            final ProjectAnalysisJob job = createAnalysisJob(space.getUniverse(), query, selection,
                    OutputFormat.JSON);
            //
            final OutputFormat outFormat;
            if (fileext == null) {
                outFormat = OutputFormat.CSV;
            } else {
                outFormat = OutputFormat.valueOf(fileext.toUpperCase());
            }

            final OutputCompression outCompression;
            if (compression == null) {
                outCompression = OutputCompression.NONE;
            } else {
                outCompression = OutputCompression.valueOf(compression.toUpperCase());
            }

            final ExportSourceWriter writer = getWriter(outFormat);
            if (writer == null) {
                throw new APIException("unable to handle the format='" + outFormat + "'", true);
            }

            StreamingOutput stream = new StreamingOutput() {
                @Override
                public void write(OutputStream os) throws IOException, WebApplicationException {
                    try {
                        if (outCompression == OutputCompression.GZIP) {
                            try {
                                os = new GZIPOutputStream(os);
                            } catch (IOException e) {
                                throw new RuntimeException(e);
                            }
                        }
                        AnalysisJobComputer.INSTANCE.compute(userContext, job, os, writer, false);
                    } catch (InterruptedException | ComputingException e) {
                        throw new IOException(e);
                    }
                }
            };

            // build the response
            ResponseBuilder response;
            String outname = filename;
            if (filename == null || filename.equals("")) {
                outname = "job-"
                        + (space.hasBookmark() ? space.getBookmark().getName() : space.getRoot().getName());
            }
            String mediaType;
            switch (outFormat) {
            case CSV:
                mediaType = "text/csv";
                outname += "." + (fileext != null ? fileext : "csv");
                break;
            case XLS:
                mediaType = "application/vnd.ms-excel";
                outname += "." + (fileext != null ? fileext : "xls");
                break;
            case XLSX:
                mediaType = "application/vnd.openxmlformats-officedocument.spreadsheetml.sheet";
                outname += "." + (fileext != null ? fileext : "xlsx");
                break;
            default:
                mediaType = MediaType.APPLICATION_JSON_TYPE.toString();
                outname += "." + (fileext != null ? fileext : "json");
            }

            switch (outCompression) {
            case GZIP:
                // note : setting "Content-Type:application/octet-stream" should
                // disable interceptor's GZIP compression.
                mediaType = MediaType.APPLICATION_OCTET_STREAM_TYPE.toString();
                outname += "." + (compression != null ? compression : "gz");
                break;
            default:
                // NONE
            }

            response = Response.ok(stream);
            // response.header("Content-Type", mediaType);
            if ((filename != null && filename.length() > 0)
                    && ((outFormat != OutputFormat.JSON) || (outCompression != OutputCompression.NONE))) {
                logger.info("returning results as " + mediaType + ", fileName : " + outname);
                response.header("Content-Disposition", "attachment; filename=" + outname);
            }

            return response.type(mediaType + "; charset=UTF-8").build();
        } catch (ComputingException | InterruptedException | ScopeException e) {
            throw new APIException(e.getMessage(), true);
        }
    }

    private ExportSourceWriter getWriter(OutputFormat outFormat) {
        if (outFormat == OutputFormat.CSV) {
            ExportSourceWriterCSV exporter = new ExportSourceWriterCSV();
            //settings.setQuotechar('\0');
            return exporter;
        } else if (outFormat == OutputFormat.XLS) {
            ExcelSettingsBean settings = new ExcelSettingsBean();
            settings.setExcelFile(ExcelFile.XLS);
            return new ExportSourceWriterXLSX(settings);
        } else if (outFormat == OutputFormat.XLSX) {
            ExcelSettingsBean settings = new ExcelSettingsBean();
            settings.setExcelFile(ExcelFile.XLSX);
            return new ExportSourceWriterXLSX(settings);
        } else {
            return null;
        }
    }

    /**
     * @param space
     * @param query
     * @param config
     * @return
     * @throws ScopeException 
     */
    private FacetSelection createFacetSelection(Space space, AnalyticsQuery query) throws ScopeException {
        FacetSelection selection = new FacetSelection();
        SpaceScope scope = new SpaceScope(space);
        Domain domain = space.getDomain();
        // handle period & timeframe
        if (query.getPeriod() != null && !query.getPeriod().equals("") && query.getTimeframe() != null
                && query.getTimeframe().size() > 0) {
            ExpressionAST expr = scope.parseExpression(query.getPeriod());
            Facet facet = createFacetInterval(space, expr, query.getTimeframe());
            selection.getFacets().add(facet);
        }
        // handle compareframe
        if (query.getPeriod() != null && !query.getPeriod().equals("") && query.getCompareTo() != null
                && query.getCompareTo().size() > 0) {
            ExpressionAST expr = scope.parseExpression(query.getPeriod());
            Facet compareFacet = createFacetInterval(space, expr, query.getCompareTo());
            selection.setCompareTo(Collections.singletonList(compareFacet));
        }
        // handle filters
        if (query.getFilters() != null) {
            Facet segment = SegmentManager.newSegmentFacet(domain);
            for (String filter : query.getFilters()) {
                filter = filter.trim();
                if (!filter.equals("")) {
                    try {
                        ExpressionAST filterExpr = scope.parseExpression(filter);
                        if (!filterExpr.getImageDomain().isInstanceOf(IDomain.CONDITIONAL)) {
                            throw new ScopeException("invalid filter, must be a condition");
                        }
                        Facet facet = createFacet(filterExpr);
                        if (facet != null) {
                            selection.getFacets().add(facet);
                        } else {
                            // use open-filter
                            FacetMemberString openFilter = SegmentManager.newOpenFilter(filterExpr, filter);
                            segment.getSelectedItems().add(openFilter);
                        }
                    } catch (ScopeException e) {
                        query.add(new Problem(Severity.ERROR, filter,
                                "invalid filter definition: \n" + e.getMessage(), e));
                    }
                }
            }
            if (!segment.getSelectedItems().isEmpty()) {
                selection.getFacets().add(segment);
            }
        }
        //
        return selection;
    }

    private Facet createFacet(ExpressionAST expr) {
        if (expr instanceof Operator) {
            Operator op = (Operator) expr;
            if (op.getOperatorDefinition().getId() == IntrinsicOperators.EQUAL & op.getArguments().size() == 2) {
                ExpressionAST dim = op.getArguments().get(0);
                ExpressionAST value = op.getArguments().get(1);
                if (value instanceof ConstantValue) {
                    Facet facet = new Facet();
                    facet.setId(dim.prettyPrint());
                    Object constant = ((ConstantValue) value).getValue();
                    if (constant != null) {
                        facet.getSelectedItems()
                                .add(new FacetMemberString(constant.toString(), constant.toString()));
                        return facet;
                    }
                }
            } else if (op.getOperatorDefinition().getId() == IntrinsicOperators.IN
                    & op.getArguments().size() == 2) {
                ExpressionAST dim = op.getArguments().get(0);
                ExpressionAST second = op.getArguments().get(1);
                if (second instanceof Operator) {
                    Operator vector = (Operator) second;
                    if (vector.getOperatorDefinition().getExtendedID().equals(VectorOperatorDefinition.ID)) {
                        Facet facet = new Facet();
                        facet.setId(dim.prettyPrint());
                        for (ExpressionAST value : vector.getArguments()) {
                            Object constant = ((ConstantValue) value).getValue();
                            if (constant != null) {
                                FacetMember member = new FacetMemberString(constant.toString(),
                                        constant.toString());
                                facet.getSelectedItems().add(member);
                            }
                        }
                        if (!facet.getSelectedItems().isEmpty()) {
                            return facet;
                        }
                    }
                }
            }
        }
        // else
        return null;
    }

    private Facet createFacetInterval(Space space, ExpressionAST expr, List<String> values) throws ScopeException {
        Facet facet = new Facet();
        facet.setId(rewriteExpressionToGlobalScope(expr, space));
        String lowerbound = values.get(0);
        String upperbound = values.size() == 2 ? values.get(1) : lowerbound;
        FacetMemberInterval member = new FacetMemberInterval(lowerbound, upperbound);
        facet.getSelectedItems().add(member);
        return facet;
    }

    private ProjectAnalysisJob createAnalysisJob(Universe universe, AnalyticsQuery query, FacetSelection selection,
            OutputFormat format) throws ScopeException {
        // read the domain reference
        if (query.getDomain() == null) {
            throw new ScopeException("incomplete specification, you must specify the data domain expression");
        }
        Domain domain = getDomain(universe, query.getDomain());
        AccessRightsUtils.getInstance().checkRole(universe.getContext(), domain, AccessRight.Role.READ);
        // the rest of the ACL is delegated to the AnalysisJob
        Space root = universe.S(domain);

        // handle the columns
        List<Metric> metrics = new ArrayList<Metric>();
        List<FacetExpression> facets = new ArrayList<FacetExpression>();
        //DomainExpressionScope domainScope = new DomainExpressionScope(universe, domain);
        int facetCount = 0;
        int legacyFacetCount = 0;// count how much real facets we have to
        // translate indexes
        int legacyMetricCount = 0;
        HashMap<Integer, Integer> lookup = new HashMap<>();// convert simple
        // indexes into
        // analysisJob
        // indexes
        HashSet<Integer> metricSet = new HashSet<>();// mark metrics
        if ((query.getGroupBy() == null || query.getGroupBy().isEmpty())
                && (query.getMetrics() == null || query.getMetrics().isEmpty())) {
            throw new ScopeException("there is no defined facet, can't run the analysis");
        }
        // now we are going to use the domain Space scope
        // -- note that it won't limit the actual expression scope to the bookmark scope - but let's keep that for latter
        SpaceScope scope = new SpaceScope(universe.S(domain));
        // add the period parameter if available
        if (query.getPeriod() != null && !query.getPeriod().equals("")) {
            try {
                ExpressionAST period = scope.parseExpression(query.getPeriod());
                scope.addParam("__PERIOD", period);
            } catch (ScopeException e) {
                // ignore
            }
        }
        // quick fix to support the old facet mechanism
        ArrayList<String> analysisFacets = new ArrayList<>();
        if (query.getGroupBy() != null)
            analysisFacets.addAll(query.getGroupBy());
        if (query.getMetrics() != null)
            analysisFacets.addAll(query.getMetrics());
        for (String facet : analysisFacets) {
            if (facet != null && facet.length() > 0) {// ignore empty values
                ExpressionAST colExpression = scope.parseExpression(facet);
                IDomain image = colExpression.getImageDomain();
                if (image.isInstanceOf(IDomain.AGGREGATE)) {
                    IDomain source = colExpression.getSourceDomain();
                    String name = colExpression.getName();// T1807
                    if (!source.isInstanceOf(DomainDomain.DOMAIN)) {
                        // need to add the domain
                        // check if it needs grouping?
                        if (colExpression instanceof Operator) {
                            Operator op = (Operator) colExpression;
                            if (op.getArguments().size() > 1 && op.getOperatorDefinition()
                                    .getPosition() != OperatorDefinition.PREFIX_POSITION) {
                                colExpression = ExpressionMaker.GROUP(colExpression);
                            }
                        }
                        // add the domain// relink with the domain
                        colExpression = ExpressionMaker.COMPOSE(new DomainReference(universe, domain),
                                colExpression);
                    }
                    // now it can be transformed into a measure
                    Measure m = universe.asMeasure(colExpression);
                    if (m == null) {
                        throw new ScopeException("cannot use expression='" + facet + "'");
                    }
                    Metric metric = new Metric();
                    metric.setExpression(new Expression(m.prettyPrint()));
                    if (name == null) {
                        name = m.getName();
                    }
                    metric.setName(name);
                    metrics.add(metric);
                    //
                    lookup.put(facetCount, legacyMetricCount++);
                    metricSet.add(facetCount);
                    facetCount++;
                } else {
                    // it's a dimension
                    IDomain source = colExpression.getSourceDomain();
                    String name = colExpression.getName();// T1807
                    if (!source.isInstanceOf(DomainDomain.DOMAIN)) {
                        // need to add the domain
                        // check if it needs grouping?
                        if (colExpression instanceof Operator) {
                            Operator op = (Operator) colExpression;
                            if (op.getArguments().size() > 1 && op.getOperatorDefinition()
                                    .getPosition() != OperatorDefinition.PREFIX_POSITION) {
                                colExpression = ExpressionMaker.GROUP(colExpression);
                            }
                        }
                        // add the domain// relink with the domain
                        colExpression = ExpressionMaker.COMPOSE(new DomainReference(universe, domain),
                                colExpression);
                    }
                    Axis axis = root.getUniverse().asAxis(colExpression);
                    if (axis == null) {
                        throw new ScopeException("cannot use expression='" + colExpression.prettyPrint() + "'");
                    }
                    if (name != null) {
                        axis.setName(name);
                    }
                    facets.add(new FacetExpression(axis.prettyPrint(), axis.getName()));
                    //
                    lookup.put(facetCount, legacyFacetCount++);
                    facetCount++;
                }
            }
        }

        // handle orderBy
        List<OrderBy> orderBy = new ArrayList<>();
        int pos = 1;
        if (query.getOrderBy() != null) {
            for (String order : query.getOrderBy()) {
                if (order != null && order.length() > 0) {
                    // let's try to parse it
                    try {
                        ExpressionAST expr = scope.parseExpression(order);
                        IDomain image = expr.getImageDomain();
                        Direction direction = getDirection(image);
                        if (image.isInstanceOf(DomainNumericConstant.DOMAIN)) {
                            // it is a reference to the facets
                            DomainNumericConstant num = (DomainNumericConstant) image
                                    .getAdapter(DomainNumericConstant.class);
                            int index = num.getValue().intValue();
                            if (!lookup.containsKey(index)) {
                                throw new ScopeException("invalid orderBy expression at position " + pos
                                        + ": the index specified (" + index + ") is out of bounds");
                            }
                            int legacy = lookup.get(index);
                            if (metricSet.contains(index)) {
                                legacy += legacyFacetCount;
                            }
                            orderBy.add(new OrderBy(legacy, direction));
                        } else {
                            // it's an expression which is now scoped into the bookmark
                            // but job is expecting it to be scoped in the universe... (OMG)
                            // also we must remove the sort operator to avoid nasty SQL error when generating the SQL
                            if (expr.getImageDomain().isInstanceOf(DomainSort.DOMAIN) && expr instanceof Operator) {
                                // remove the first operator
                                Operator op = (Operator) expr;
                                if (op.getArguments().size() == 1 && (op.getOperatorDefinition()
                                        .getExtendedID() == SortOperatorDefinition.ASC_ID
                                        || op.getOperatorDefinition()
                                                .getExtendedID() == SortOperatorDefinition.DESC_ID)) {
                                    expr = op.getArguments().get(0);
                                }
                            }
                            String universalExpression = rewriteExpressionToGlobalScope(expr, root);
                            orderBy.add(new OrderBy(new Expression(universalExpression), direction));
                        }
                    } catch (ScopeException e) {
                        throw new ScopeException(
                                "unable to parse orderBy expression at position " + pos + ": " + e.getCause(), e);
                    }
                }
                pos++;
            }
        }
        // handle rollup - fix indexes
        pos = 1;
        if (query.getRollups() != null) {
            for (RollUp rollup : query.getRollups()) {
                if (rollup != null && rollup.getCol() > -1) {// ignore grand-total
                    // can't rollup on metric
                    if (metricSet.contains(rollup.getCol())) {
                        throw new ScopeException(
                                "invalid rollup expression at position " + pos + ": the index specified ("
                                        + rollup.getCol() + ") is not valid: cannot rollup on metric");
                    }
                    if (!lookup.containsKey(rollup.getCol())) {
                        throw new ScopeException("invalid rollup expression at position " + pos
                                + ": the index specified (" + rollup.getCol() + ") is out of bounds");
                    }
                    int legacy = lookup.get(rollup.getCol());
                    rollup.setCol(legacy);
                }
            }
        }

        // create the actual job
        // - using the AnalysisQuery.getQueryID() as the job OID: this one is unique for a given query
        ProjectAnalysisJobPK pk = new ProjectAnalysisJobPK(universe.getProject().getId(), query.getQueryID());
        ProjectAnalysisJob analysisJob = new ProjectAnalysisJob(pk);
        analysisJob.setDomains(Collections.singletonList(domain.getId()));
        analysisJob.setMetricList(metrics);
        analysisJob.setFacets(facets);
        analysisJob.setOrderBy(orderBy);
        analysisJob.setSelection(selection);
        analysisJob.setRollups(query.getRollups());
        analysisJob.setAutoRun(true);

        // automatic limit?
        if (query.getLimit() == null && format == OutputFormat.JSON) {
            int complexity = analysisJob.getFacets().size();
            if (complexity < 4) {
                analysisJob.setLimit((long) Math.pow(10, complexity + 1));
            } else {
                analysisJob.setLimit(100000L);
            }
        } else {
            analysisJob.setLimit(query.getLimit());
        }

        // offset
        if (query.getOffset() != null) {
            analysisJob.setOffset(query.getOffset());
        }

        // beyond limit
        if (query.getBeyondLimit() != null && !query.getBeyondLimit().isEmpty()) {
            ArrayList<Index> indexes = new ArrayList<>(query.getBeyondLimit().size());
            for (String value : query.getBeyondLimit()) {
                // check if it is a number
                Integer x = getIntegerValue(value);
                if (x == null || x < 0 && x >= query.getGroupBy().size()) {
                    x = query.getGroupBy().indexOf(value);
                }
                if (x == null || x < 0) {
                    throw new ScopeException("invalid beyondLimit parameter: " + value
                            + ": must be an valid integer position or a groupBy expression");
                }
                indexes.add(new Index(x));
            }
            analysisJob.setBeyondLimit(indexes);
        }
        return analysisJob;
    }

    private Integer getIntegerValue(String value) {
        try {
            return Integer.parseInt(value);
        } catch (NumberFormatException e) {
            return null;
        }
    }

    private Domain getDomain(Universe universe, String definiiton) throws ScopeException {
        // -- using the universe scope for now; will change when merge with T821
        // -- to also support query
        UniverseScope scope = new UniverseScope(universe);
        ExpressionAST domainExpression = scope.parseExpression(definiiton);
        if (!(domainExpression instanceof SpaceExpression)) {
            throw new ScopeException("invalid specification, the domain expression must resolve to a Space");
        }
        Space ref = ((SpaceExpression) domainExpression).getSpace();
        Domain domain = ref.getDomain();
        return domain;
    }

    private Direction getDirection(IDomain domain) {
        if (domain.isInstanceOf(DomainSort.DOMAIN)) {
            DomainSort sort = (DomainSort) domain.getAdapter(DomainSort.class);
            if (sort != null) {
                SortDirection direction = sort.getDirection();
                if (direction != null) {
                    switch (direction) {
                    case ASC:
                        return Direction.ASC;
                    case DESC:
                        return Direction.DESC;
                    }
                }
            }
        }
        // else
        // no desc | asc operator provided: use default
        if (domain.isInstanceOf(IDomain.NUMERIC) || domain.isInstanceOf(IDomain.TEMPORAL)) {
            return Direction.DESC;
        } else { //if (image.isInstanceOf(IDomain.STRING)) {
            return Direction.ASC;
        }
    }

    private BookmarkConfig createBookmarkConfig(Space space, AnalyticsQuery query) throws ScopeException {
        SpaceScope scope = new SpaceScope(space);
        BookmarkConfig config = new BookmarkConfig();
        // config use the Domain OID
        config.setDomain(space.getDomain().getOid());
        //config.setSelection();
        config.setLimit(query.getLimit());
        if (query.getGroupBy() != null) {
            List<String> chosenDimensions = new ArrayList<>();
            for (String facet : query.getGroupBy()) {
                // add the domain scope
                ExpressionAST expr = scope.parseExpression(facet);
                chosenDimensions.add(rewriteExpressionToGlobalScope(expr, space));
            }
            String[] toArray = new String[chosenDimensions.size()];
            config.setChosenDimensions(chosenDimensions.toArray(toArray));
        }
        if (query.getMetrics() != null) {
            List<String> choosenMetrics = new ArrayList<>();
            for (String facet : query.getMetrics()) {
                // add the domain scope
                ExpressionAST expr = scope.parseExpression(facet);
                choosenMetrics.add(rewriteExpressionToGlobalScope(expr, space));
            }
            String[] toArray = new String[choosenMetrics.size()];
            config.setChosenMetrics(choosenMetrics.toArray(toArray));
        }
        //
        if (query.getOrderBy() != null) {
            config.setOrderBy(new ArrayList<OrderBy>());
            for (String orderBy : query.getOrderBy()) {
                ExpressionAST expr = scope.parseExpression(orderBy);
                Direction direction = getDirection(expr.getImageDomain());
                OrderBy copy = new OrderBy(new Expression(rewriteExpressionToGlobalScope(expr, space)), direction);
                config.getOrderBy().add(copy);
            }
        }
        //
        if (query.getRollups() != null) {
            config.setRollups(query.getRollups());
        }
        // add the selection
        FacetSelection selection = createFacetSelection(space, query);
        config.setSelection(selection);
        return config;
    }

    /**
     * merge the bookmark config with the current query. It modifies the query. Query parameters take precedence over the bookmark config.
     * 
     * @param space
     * @param query
     * @param config
     * @throws ScopeException
     * @throws ComputingException
     * @throws InterruptedException
     */
    private void mergeBoomarkConfig(Space space, AnalyticsQuery query, BookmarkConfig config)
            throws ScopeException, ComputingException, InterruptedException {
        ReferenceStyle prettyStyle = getReferenceStyle(query.getStyle());
        PrettyPrintOptions globalOptions = new PrettyPrintOptions(prettyStyle, null);
        UniverseScope globalScope = new UniverseScope(space.getUniverse());
        PrettyPrintOptions localOptions = new PrettyPrintOptions(prettyStyle, space.getTop().getImageDomain());
        SpaceScope localScope = new SpaceScope(space);
        if (query.getDomain() == null) {
            query.setDomain(space.prettyPrint(globalOptions));
        }
        if (query.getLimit() == null) {
            if (config != null) {
                query.setLimit(config.getLimit());
            }
        }
        //
        // handling the period
        //
        if (query.getPeriod() == null && config != null && config.getPeriod() != null
                && !config.getPeriod().isEmpty()) {
            // look for this domain period
            String domainID = space.getDomain().getOid();
            String period = config.getPeriod().get(domainID);
            if (period != null) {
                ExpressionAST expr = globalScope.parseExpression(period);
                IDomain image = expr.getImageDomain();
                if (image.isInstanceOf(IDomain.TEMPORAL)) {
                    // ok, it's a date
                    query.setPeriod(expr.prettyPrint(localOptions));
                }
            }
        }
        if (query.getPeriod() == null) {
            DomainHierarchy hierarchy = DomainHierarchyManager.INSTANCE
                    .getHierarchy(space.getUniverse().getProject().getId(), space.getDomain(), false);
            for (DimensionIndex index : hierarchy.getDimensionIndexes()) {
                if (index.isVisible() && index.getDimension().getType().equals(Type.CONTINUOUS)
                        && index.getAxis().getDefinitionSafe().getImageDomain().isInstanceOf(IDomain.TEMPORAL)) {
                    // use it as period
                    Axis axis = index.getAxis();
                    ExpressionAST expr = new AxisExpression(axis);
                    query.setPeriod(expr.prettyPrint(localOptions));
                    if (query.getTimeframe() == null) {
                        if (index.getStatus() == Status.DONE) {
                            query.setTimeframe(Collections.singletonList("__CURRENT_MONTH"));
                        } else {
                            query.setTimeframe(Collections.singletonList("__ALL"));
                        }
                    }
                    // quit the loop!
                    break;
                }
            }
            if (query.getPeriod() == null) {
                // nothing selected - double check and auto detect?
                if (query.getTimeframe() != null && query.getTimeframe().size() > 0) {
                    query.add(new Problem(Severity.WARNING, "period",
                            "No period defined: you cannot set the timeframe"));
                }
                if (query.getCompareTo() != null && query.getCompareTo().size() > 0) {
                    query.add(new Problem(Severity.WARNING, "period",
                            "No period defined: you cannot set the compareTo"));
                }
            }
        }
        //
        // merging groupBy
        //
        boolean groupbyWildcard = isWildcard(query.getGroupBy());
        if (query.getGroupBy() == null || groupbyWildcard) {
            List<String> groupBy = new ArrayList<String>();
            if (config == null) {
                // it is not a bookmark, then we will provide default select *
                // only if there is nothing selected at all (groupBy & metrics)
                // or user ask for it explicitly is wildcard
                if (groupbyWildcard || query.getMetrics() == null) {
                    boolean periodIsSet = false;
                    if (query.getPeriod() != null) {
                        groupBy.add(query.getPeriod());
                        periodIsSet = true;
                    }
                    // use a default pivot selection...
                    // -- just list the content of the table
                    for (Dimension dimension : space.getDimensions()) {
                        Axis axis = space.A(dimension);
                        try {
                            DimensionIndex index = axis.getIndex();
                            IDomain image = axis.getDefinitionSafe().getImageDomain();
                            if (index != null && index.isVisible() && index.getStatus() != Status.ERROR
                                    && !image.isInstanceOf(IDomain.OBJECT)) {
                                boolean isTemporal = image.isInstanceOf(IDomain.TEMPORAL);
                                if (!isTemporal || !periodIsSet) {
                                    groupBy.add(axis.prettyPrint(localOptions));
                                    if (isTemporal)
                                        periodIsSet = true;
                                }
                            }
                        } catch (ComputingException | InterruptedException e) {
                            // ignore this one
                        }
                    }
                }
            } else if (config.getChosenDimensions() != null) {
                for (String chosenDimension : config.getChosenDimensions()) {
                    try {
                        String f = null;
                        if (chosenDimension.startsWith("@")) {
                            // need to fix the scope
                            ExpressionAST expr = globalScope.parseExpression(chosenDimension);
                            f = expr.prettyPrint(localOptions);//rewriteExpressionToLocalScope(expr, space);
                        } else {
                            // legacy support raw ID
                            // parse to validate and apply prettyPrint options
                            ExpressionAST expr = localScope.parseExpression("@'" + chosenDimension + "'");
                            f = expr.prettyPrint(localOptions);
                        }
                        groupBy.add(f);
                    } catch (ScopeException e) {
                        query.add(new Problem(Severity.WARNING, chosenDimension,
                                "failed to parse bookmark dimension: " + e.getMessage(), e));
                    }
                }
            }
            if (groupbyWildcard) {
                query.getGroupBy().remove(0);// remove the first one
                groupBy.addAll(query.getGroupBy());// add reminding
            }
            query.setGroupBy(groupBy);
        }
        // merging Metrics
        boolean metricWildcard = isWildcard(query.getMetrics());
        if (query.getMetrics() == null || metricWildcard) {
            List<String> metrics = new ArrayList<>();
            if (config == null) {
                boolean someIntrinsicMetric = false;
                for (Measure measure : space.M()) {
                    Metric metric = measure.getMetric();
                    if (metric != null && !metric.isDynamic()) {
                        IDomain image = measure.getDefinitionSafe().getImageDomain();
                        if (image.isInstanceOf(IDomain.AGGREGATE)) {
                            Measure m = space.M(metric);
                            metrics.add((new MeasureExpression(m)).prettyPrint(localOptions));
                            //metrics.add(rewriteExpressionToLocalScope(new MeasureExpression(m), space));
                            someIntrinsicMetric = true;
                        }
                    }
                }
                if (!someIntrinsicMetric) {
                    metrics.add("count() // default metric");
                }
            } else if (config.getChosenMetrics() != null) {
                for (String chosenMetric : config.getChosenMetrics()) {
                    // parse to validate and reprint
                    try {
                        ExpressionAST expr = localScope.parseExpression("@'" + chosenMetric + "'");
                        metrics.add(expr.prettyPrint(localOptions));
                    } catch (ScopeException e) {
                        query.add(new Problem(Severity.WARNING, chosenMetric,
                                "failed to parse bookmark metric: " + e.getMessage(), e));
                    }
                }
            } else if (config.getAvailableMetrics() != null
                    && (query.getGroupBy() == null || query.getGroupBy().isEmpty())) {
                // no axis selected, no choosen metrics, but available metrics
                // this is an old bookmark (analytics), that used to display the KPIs
                // so just compute the KPIs
                for (String availableMetric : config.getAvailableMetrics()) {
                    // parse to validate and reprint
                    try {
                        ExpressionAST expr = localScope.parseExpression("@'" + availableMetric + "'");
                        metrics.add(expr.prettyPrint(localOptions));
                    } catch (ScopeException e) {
                        query.add(new Problem(Severity.WARNING, availableMetric,
                                "failed to parse bookmark metric: " + e.getMessage(), e));
                    }
                }
            }
            if (metricWildcard) {
                query.getMetrics().remove(0);// remove the first one
                metrics.addAll(query.getMetrics());// add reminding
            }
            query.setMetrics(metrics);
        }
        if (query.getOrderBy() == null) {
            if (config != null && config.getOrderBy() != null) {
                query.setOrderBy(new ArrayList<String>());
                for (OrderBy orderBy : config.getOrderBy()) {
                    // legacy issue? in some case the bookmark contains invalid orderBy expressions
                    if (orderBy.getExpression() != null) {
                        ExpressionAST expr = globalScope.parseExpression(orderBy.getExpression().getValue());
                        IDomain image = expr.getImageDomain();
                        if (!image.isInstanceOf(DomainSort.DOMAIN)) {
                            if (orderBy.getDirection() == Direction.ASC) {
                                expr = ExpressionMaker.ASC(expr);
                            } else {
                                expr = ExpressionMaker.DESC(expr);
                            }
                        }
                        query.getOrderBy().add(expr.prettyPrint(localOptions));
                    }
                }
            }
        }
        if (query.getRollups() == null) {
            if (config != null) {
                query.setRollups(config.getRollups());
            }
        }
        //
        // handling selection
        //
        FacetSelection selection = config != null ? config.getSelection() : new FacetSelection();
        boolean filterWildcard = isWildcardFilters(query.getFilters());
        List<String> filters = query.getFilters() != null ? new ArrayList<>(query.getFilters())
                : new ArrayList<String>();
        if (filterWildcard) {
            filters.remove(0); // remove the *
        }
        if (!selection.getFacets().isEmpty()) {// always iterate over selection at least to capture the period
            boolean keepConfig = filterWildcard || filters.isEmpty();
            String period = null;
            if (query.getPeriod() != null && query.getTimeframe() == null) {
                ExpressionAST expr = localScope.parseExpression(query.getPeriod());
                period = expr.prettyPrint(new PrettyPrintOptions(ReferenceStyle.IDENTIFIER, null));
            }
            // look for the selection
            for (Facet facet : selection.getFacets()) {
                if (!facet.getSelectedItems().isEmpty()) {
                    if (facet.getId().equals(period)) {
                        // it's the period
                        List<FacetMember> items = facet.getSelectedItems();
                        if (items.size() == 1) {
                            FacetMember timeframe = items.get(0);
                            if (timeframe instanceof FacetMemberInterval) {
                                String upperBound = ((FacetMemberInterval) timeframe).getUpperBound();
                                if (upperBound.startsWith("__")) {
                                    // it's a shortcut
                                    query.setTimeframe(Collections.singletonList(upperBound));
                                } else {
                                    // it's a date
                                    String lowerBound = ((FacetMemberInterval) timeframe).getLowerBound();
                                    query.setTimeframe(new ArrayList<String>(2));
                                    query.getTimeframe().add(lowerBound);
                                    query.getTimeframe().add(upperBound);
                                }
                            }
                        }
                    } else if (SegmentManager.isSegmentFacet(facet) && keepConfig) {
                        // it's the segment facet
                        for (FacetMember item : facet.getSelectedItems()) {
                            if (item instanceof FacetMemberString) {
                                FacetMemberString member = (FacetMemberString) item;
                                if (SegmentManager.isOpenFilter(member)) {
                                    // open filter is jut a formula
                                    String formula = member.getValue();
                                    if (formula.startsWith("=")) {
                                        formula = formula.substring(1);
                                    }
                                    filters.add(formula);
                                } else {
                                    // it's a segment name
                                    // check the ID
                                    try {
                                        if (member.getId().startsWith("@")) {
                                            ExpressionAST seg = globalScope.parseExpression(member.getId());
                                            filters.add(seg.prettyPrint(localOptions));
                                        } else {
                                            // use the name
                                            ExpressionAST seg = globalScope
                                                    .parseExpression("'" + member.getValue() + "'");
                                            filters.add(seg.prettyPrint(localOptions));
                                        }
                                    } catch (ScopeException e) {
                                        query.add(new Problem(Severity.ERROR, member.getId(),
                                                "Unable to parse segment with value='" + member + "'", e));
                                    }
                                }
                            }
                        }
                    } else if (keepConfig) {
                        ExpressionAST expr = globalScope.parseExpression(facet.getId());
                        String filter = expr.prettyPrint(localOptions);
                        if (facet.getSelectedItems().size() == 1) {
                            if (facet.getSelectedItems().get(0) instanceof FacetMemberString) {
                                filter += "=";
                                FacetMember member = facet.getSelectedItems().get(0);
                                filter += "\"" + member.toString() + "\"";
                                filters.add(filter);
                            }
                        } else {
                            filter += " IN {";
                            boolean first = true;
                            for (FacetMember member : facet.getSelectedItems()) {
                                if (member instanceof FacetMemberString) {
                                    if (!first) {
                                        filter += " , ";
                                    } else {
                                        first = false;
                                    }
                                    filter += "\"" + member.toString() + "\"";
                                }
                            }
                            filter += "}";
                            if (!first) {
                                filters.add(filter);
                            }
                        }
                    }
                }
            }
        }
        query.setFilters(filters);
        //
        // check timeframe again
        if (query.getPeriod() != null && (query.getTimeframe() == null || query.getTimeframe().size() == 0)) {
            // add a default timeframe
            query.setTimeframe(Collections.singletonList("__CURRENT_MONTH"));
        }
    }

    private PrettyPrintOptions.ReferenceStyle getReferenceStyle(Style style) {
        switch (style) {
        case HUMAN:
        case HTML:
            return ReferenceStyle.NAME;
        case LEGACY:
            return ReferenceStyle.LEGACY;
        case ROBOT:
        default:
            return ReferenceStyle.IDENTIFIER;
        }
    }

    private boolean isWildcard(List<String> facets) {
        if (facets != null && !facets.isEmpty()) {
            String first = facets.get(0);
            return first.equals("*");
        }
        // else
        return false;
    }

    private boolean isWildcardFilters(List<String> items) {
        if (items != null && !items.isEmpty()) {
            String first = items.get(0);
            return first.equals("*");
        } else {
            return false;// only return true if it is a real wildcard
        }
    }

    /**
     * rewrite a local expression valid in the root scope as a global expression
     * @param expr
     * @param root
     * @return
     * @throws ScopeException
     */
    private String rewriteExpressionToGlobalScope(ExpressionAST expr, Space root) throws ScopeException {
        IDomain source = expr.getSourceDomain();
        if (!source.isInstanceOf(DomainDomain.DOMAIN)) {
            String global = root.prettyPrint();
            String value = expr.prettyPrint();
            return global + ".(" + value + ")";
        } else {
            return expr.prettyPrint();
        }
    }

    /**
     * @param uriInfo 
     * @param userContext
     * @param bBID
     * @param x
     * @param y
     * @param color
     * @param style 
     * @param query
     * @return
     * @throws InterruptedException 
     * @throws ComputingException 
     * @throws ScopeException 
     */
    public Response viewAnalysis(final AppContext userContext, String BBID, ViewQuery view, String data,
            Style style, String envelope) throws ScopeException, ComputingException, InterruptedException {
        Space space = getSpace(userContext, BBID);
        //
        if (data == null)
            data = style == Style.HTML ? "EMBEDED" : "URL";
        boolean preFetch = true;// default to prefetch when data mode is URL
        //
        Bookmark bookmark = space.getBookmark();
        BookmarkConfig config = BookmarkManager.INSTANCE.readConfig(bookmark);
        //
        // handle the limit
        Long explicitLimit = view.getLimit();
        AnalyticsQueryImpl query = new AnalyticsQueryImpl(view);
        // merge the bookmark config with the query
        mergeBoomarkConfig(space, query, config);
        //
        // change the query ref to use the domain one
        // - we don't want to have side-effects
        String domainBBID = "@'" + space.getUniverse().getProject().getOid() + "'.@'" + space.getDomain().getOid()
                + "'";
        query.setBBID(domainBBID);
        //
        VegaliteConfigurator inputConfig = new VegaliteConfigurator(space, query);
        // first check the provided parameters, because they must override the default
        inputConfig.createChannelDef("x", view.getX());
        inputConfig.createChannelDef("y", view.getY());
        inputConfig.createChannelDef("color", view.getColor());
        inputConfig.createChannelDef("size", view.getSize());
        inputConfig.createChannelDef("column", view.getColumn());
        inputConfig.createChannelDef("row", view.getRow());
        if (inputConfig.getRequired().getMetrics().size() > 0) {
            // override the default metrics
            query.setMetrics(inputConfig.getRequired().getMetrics());
        }
        if (inputConfig.getRequired().getGroupBy().size() > 0) {
            // override the default metrics
            query.setGroupBy(inputConfig.getRequired().getGroupBy());
        }
        //
        // add the compareTo() if needed
        /*
        if (query.getCompareframe()!=null && query.getMetrics()!=null && query.getMetrics().size()==1) {
           String expression = query.getMetrics().get(0);
           ExpressionAST ast = inputConfig.parse(expression);
           query.getMetrics().add("compareTo("+inputConfig.prettyPrint(ast)+")");
        }
        */
        //
        int dims = (query.getGroupBy() != null) ? query.getGroupBy().size() : 0;
        int kpis = (query.getMetrics() != null) ? query.getMetrics().size() : 0;
        // this will be the output config including the default settings
        VegaliteConfigurator outputConfig = new VegaliteConfigurator(space, query);
        //
        // use default dataviz unless x and y are already sets
        if (view.getX() == null || view.getY() == null) {
            // DOMAIN
            if (config == null) {
                // not a bookmark, use default & specs if nothing provided
                // T1935: if explicit groupBy, use it
                if (view.getGroupBy() != null && view.getGroupBy().size() > 0 && query.getGroupBy() != null
                        && query.getGroupBy().size() > 0) {
                    int next = 0;
                    while (next < dims) {
                        if (!inputConfig.getRequired().getGroupBy().contains(query.getGroupBy().get(next))) {
                            if (view.getX() == null) {
                                // use it as the x
                                view.setX(query.getGroupBy().get(next++));
                            } else if (view.getColor() == null) {
                                // use it as the color
                                view.setColor(query.getGroupBy().get(next++));
                            } else if (view.getColumn() == null) {
                                // use it as the column
                                view.setColumn(query.getGroupBy().get(next++));
                            } else if (view.getRow() == null) {
                                // use it as the column
                                view.setRow(query.getGroupBy().get(next++));
                            } else {
                                break;// no more channel available
                            }
                        }
                    }
                }
                if (view.getX() == null && !inputConfig.isTimeseries() && query.getPeriod() != null) {
                    view.setX("__PERIOD");
                }
                if (query.getMetrics() == null || query.getMetrics().size() == 0) {
                    if (!inputConfig.isHasMetric()) {
                        // use count() for now
                        if (view.getX() == null) {
                            view.setX("count()");
                        } else {
                            view.setY("count()");
                        }
                    }
                } else {
                    // display some metrics
                    // - single metric
                    if (query.getMetrics().size() == 1) {
                        if (!inputConfig.isHasMetric()) {
                            if (view.getX() == null) {
                                view.setX(query.getMetrics().get(0));
                            } else {
                                view.setY(query.getMetrics().get(0));
                            }
                        }
                        // - multiple metrics
                    } else if (view.getY() == null || view.getColor() == null || view.getColumn() == null
                            || view.getRow() == null) {
                        // set __VALUE
                        if (!inputConfig.isHasMetricValue()) {
                            if (view.getX() == null) {
                                view.setX("__VALUE");
                            } else {
                                view.setY("__VALUE");
                            }
                        }
                        // set __METRICS
                        if (!inputConfig.isHasMetricSeries()) {
                            if (view.getY() == null) {
                                view.setY("__METRICS");
                            } else if (view.getColor() == null) {
                                view.setColor("__METRICS");
                            } else if (view.getColumn() == null) {
                                view.setColumn("__METRICS");
                            } else if (view.getRow() == null) {
                                view.setRow("__METRICS");
                            }
                        }
                    }
                }
                // TIME-SERIES
            } else if (config.getCurrentAnalysis() != null
                    && config.getCurrentAnalysis().equalsIgnoreCase(BookmarkConfig.TIMESERIES_ANALYSIS)) {
                // use the period as the x
                if (view.getX() == null && !inputConfig.isTimeseries()) {
                    view.setX("__PERIOD");
                }
                // take care of y axis and metrics
                if (view.getY() == null && (!inputConfig.isHasMetric() || !inputConfig.isHasMetricValue())) {
                    if (kpis == 0 && !inputConfig.isHasMetric()) {
                        // we need a default metric
                        view.setY("count() // this is the default metric");
                    } else if (kpis == 1 && !inputConfig.isHasMetric()) {
                        // we can only use the first one for now
                        view.setY(query.getMetrics().get(0));
                    } else if (!inputConfig.isHasMetricValue()) {
                        view.setY(TransposeConverter.METRIC_VALUE_COLUMN);
                        if (!inputConfig.isHasMetricSeries()) {
                            if (view.getColor() == null) {
                                view.setColor(TransposeConverter.METRIC_SERIES_COLUMN);
                            } else if (view.getColumn() == null) {
                                view.setColumn(TransposeConverter.METRIC_SERIES_COLUMN);
                            } else if (view.getRow() == null) {
                                view.setRow(TransposeConverter.METRIC_SERIES_COLUMN);
                            }
                        }
                    }
                }
                // add reminding groupBy
                int next = 0;
                while (next < dims) {
                    String groupBy = query.getGroupBy().get(next++);
                    if (!groupBy.equals("__PERIOD") && !groupBy.equals(query.getPeriod())
                            && !inputConfig.getRequired().getGroupBy().contains(groupBy)) {
                        if (view.getColor() == null) {
                            // use it as the color
                            view.setColor(groupBy);
                        } else if (view.getColumn() == null) {
                            // use it as the column
                            view.setColumn(groupBy);
                        } else if (view.getRow() == null) {
                            // use it as the column
                            view.setRow(groupBy);
                        } else {
                            break;// no more channel available
                        }
                    }
                }
                // BARCHART 
            } else if (config.getCurrentAnalysis() != null
                    && config.getCurrentAnalysis().equalsIgnoreCase(BookmarkConfig.BARCHART_ANALYSIS)) {
                if (view.getY() == null && (!inputConfig.isHasMetric() || !inputConfig.isHasMetricValue())) {
                    if (kpis == 0 && !inputConfig.isHasMetric()) {
                        // we need a default metric
                        view.setY("count() // this is the default metric");
                    } else if (kpis == 1 && !inputConfig.isHasMetric()) {
                        // we can only use the first one for now
                        view.setY(query.getMetrics().get(0));
                    } else if (!inputConfig.isHasMetricValue()) {
                        view.setY(TransposeConverter.METRIC_VALUE_COLUMN);
                        if (!inputConfig.isHasMetricSeries()) {
                            if (view.getColor() == null) {
                                view.setColor(TransposeConverter.METRIC_SERIES_COLUMN);
                            } else if (view.getColumn() == null) {
                                view.setColumn(TransposeConverter.METRIC_SERIES_COLUMN);
                            } else if (view.getRow() == null) {
                                view.setRow(TransposeConverter.METRIC_SERIES_COLUMN);
                            }
                        }
                    }
                }
                // add reminding groupBy
                int next = 0;
                while (next < dims) {
                    if (!inputConfig.getRequired().getGroupBy().contains(query.getGroupBy().get(next))) {
                        if (view.getX() == null) {
                            // use it as the x
                            view.setX(query.getGroupBy().get(next++));
                        } else if (view.getColor() == null) {
                            // use it as the color
                            view.setColor(query.getGroupBy().get(next++));
                        } else if (view.getColumn() == null) {
                            // use it as the column
                            view.setColumn(query.getGroupBy().get(next++));
                        } else if (view.getRow() == null) {
                            // use it as the column
                            view.setRow(query.getGroupBy().get(next++));
                        } else {
                            break;// no more channel available
                        }
                    }
                }
            } else {// TABLE_ANALYSIS or unknown
                if (kpis == 0) {
                    // we need at least one dim
                    query.setMetrics(Collections.singletonList("count() // this is the default metric"));
                    kpis++;
                }
                if (dims == 0) {
                    // just display the metrics
                    if (view.getX() == null && !inputConfig.isHasMetric() && !inputConfig.isHasMetricValue()) {
                        if (kpis == 1) {
                            view.setX(query.getMetrics().get(0));
                        } else {
                            // multi-kpis
                            view.setX(TransposeConverter.METRIC_VALUE_COLUMN);
                            if (!inputConfig.isHasMetricSeries()) {
                                view.setY(TransposeConverter.METRIC_SERIES_COLUMN);
                            }
                        }
                    }
                } else if (kpis == 1) {
                    // display a barchart or timeseries
                    if (view.getY() == null && !inputConfig.isHasMetric() && !inputConfig.isHasMetricValue()) {
                        view.setY(query.getMetrics().get(0));
                    }
                    for (String next : query.getGroupBy()) {
                        if (view.getX() == null) {
                            if (!next.equals(query.getPeriod())) {
                                // change the barchart orientation
                                view.setX(view.getY());
                                view.setY(next);
                            } else {
                                view.setX(next);
                            }
                        } else if (view.getColor() == null) {
                            // use it as the column
                            view.setColor(next);
                        } else if (view.getColumn() == null) {
                            // use it as the column
                            view.setColumn(next);
                        } else if (view.getRow() == null) {
                            // use it as the column
                            view.setRow(next);
                        } else {
                            break;// no more channel available
                        }
                    }
                } else {
                    // multiple kpis
                    if (view.getX() == null) {
                        view.setX(query.getGroupBy().get(0));
                    }
                    if (view.getY() == null && !inputConfig.isHasMetric()) {
                        if (!inputConfig.isHasMetricValue()) {
                            view.setY(TransposeConverter.METRIC_VALUE_COLUMN);
                        }
                        if (!inputConfig.isHasMetricSeries()) {
                            if (view.getColor() == null) {
                                view.setColor(TransposeConverter.METRIC_SERIES_COLUMN);
                            } else if (view.getColumn() == null) {
                                view.setColumn(TransposeConverter.METRIC_SERIES_COLUMN);
                            } else if (view.getRow() == null) {
                                view.setRow(TransposeConverter.METRIC_SERIES_COLUMN);
                            }
                        }
                    }
                    int next = 1;
                    while (next < dims) {
                        if (view.getColumn() == null) {
                            // use it as the column
                            view.setColumn(query.getGroupBy().get(next++));
                        } else if (view.getRow() == null) {
                            // use it as the column
                            view.setRow(query.getGroupBy().get(next++));
                        } else {
                            break;// no more channel available
                        }
                    }
                }
            }
        }
        // rollup is not supported
        if (query.getRollups() != null) {
            query.setRollups(null);
        }
        //
        //
        VegaliteSpecs specs = new VegaliteSpecs();
        specs.encoding.x = outputConfig.createChannelDef("x", view.getX());
        specs.encoding.y = outputConfig.createChannelDef("y", view.getY());
        specs.encoding.color = outputConfig.createChannelDef("color", view.getColor());
        specs.encoding.size = outputConfig.createChannelDef("size", view.getSize());
        specs.encoding.column = outputConfig.createChannelDef("column", view.getColumn());
        specs.encoding.row = outputConfig.createChannelDef("row", view.getRow());
        //
        if (specs.encoding.x.type == DataType.nominal && specs.encoding.y.type == DataType.quantitative) {
            // auto sort
            specs.encoding.x.sort = new Sort(specs.encoding.y.field, Operation.max, Order.descending);
        } else if (specs.encoding.y.type == DataType.nominal && specs.encoding.x.type == DataType.quantitative) {
            // auto sort
            specs.encoding.y.sort = new Sort(specs.encoding.x.field, Operation.max, Order.descending);
        }
        //
        // force using required
        query.setGroupBy(outputConfig.getRequired().getGroupBy());
        query.setMetrics(outputConfig.getRequired().getMetrics());
        //
        // enforce the explicit limit
        if (explicitLimit == null) {// compute the default
            int validDimensions = dims;
            if (outputConfig.isTimeseries())
                validDimensions--;
            if (outputConfig.isHasMetricSeries())
                validDimensions--;// excluding the metrics series
            if (dims > 0) {
                explicitLimit = 10L;// keep 10 for each dim
                for (int i = validDimensions - 1; i > 0; i--)
                    explicitLimit = explicitLimit * 10;// get the power
            }
        }
        if (explicitLimit != null && // if time-series, there's not explicit limit
                (query.getLimit() == null || query.getLimit() > explicitLimit)) {
            query.setLimit(explicitLimit);
        }
        // beyond limit
        if (outputConfig.isTimeseries()) {
            query.setBeyondLimit(Collections.singletonList(Integer.toString(outputConfig.getTimeseriesPosition())));
            query.setMaxResults(null);
        } else {
            /*
            // can add page size
            if (query.getLimit()>10 && query.getMaxResults()==null) {
               query.setMaxResults(10);
            }
            */
        }
        final int startIndex = query.getStartIndex() != null ? query.getStartIndex() : 0;
        final int maxResults = query.getMaxResults() != null ? query.getMaxResults() : query.getLimit().intValue();
        // make sure we order by something
        if (query.getOrderBy() == null || query.getOrderBy().size() == 0) {
            if (query.getMetrics().size() > 0) {
                ExpressionAST m = outputConfig.parse(query.getMetrics().get(0));
                query.setOrderBy(Collections.singletonList("desc(" + outputConfig.prettyPrint(m) + ")"));
            } else {
                query.setOrderBy(Collections.singletonList("desc(count())"));
            }
        } else {
            // check orderBy
            if (query.getMetrics().size() > 0) {
                ExpressionAST m = outputConfig.parse(query.getMetrics().get(0));
                boolean check = false;
                for (String orderBy : query.getOrderBy()) {
                    ExpressionAST o = outputConfig.parse(orderBy);
                    if (o.getImageDomain().isInstanceOf(DomainSort.DOMAIN) && o instanceof Operator) {
                        // remove the first operator
                        Operator op = (Operator) o;
                        if (op.getArguments().size() == 1 && (op.getOperatorDefinition()
                                .getExtendedID() == SortOperatorDefinition.ASC_ID
                                || op.getOperatorDefinition().getExtendedID() == SortOperatorDefinition.DESC_ID)) {
                            o = op.getArguments().get(0);
                        }
                    }
                    if (o.equals(m)) {
                        check = true;
                    }
                }
                if (!check) {
                    query.getOrderBy().add(0, "desc(" + outputConfig.prettyPrint(m) + ")");
                }
            }
        }
        //
        // create the facet selection
        FacetSelection selection = createFacetSelection(space, query);
        final ProjectAnalysisJob job = createAnalysisJob(space.getUniverse(), query, selection, OutputFormat.JSON);
        //
        // handling data
        AnalyticsResult.Info info = null;
        if (data.equals("EMBEDED")) {
            DataMatrix matrix = compute(userContext, job, query.getMaxResults(), query.getStartIndex(), false);
            if (!outputConfig.isHasMetricSeries()) {
                specs.data = transformToVegaData(query, matrix, "RECORDS");
            } else {
                specs.data = transformToVegaData(query, matrix, "TRANSPOSE");
            }
            int end = startIndex + maxResults;
            if (end > matrix.getRows().size())
                end = matrix.getRows().size();
            info = getAnalyticsResultInfo(end - startIndex, startIndex, matrix);
        } else if (data.equals("URL")) {
            if (preFetch || style == Style.HTML) {// always prefetch if HTML
                // run the query
                Callable<DataMatrix> task = new Callable<DataMatrix>() {
                    @Override
                    public DataMatrix call() throws Exception {
                        return compute(userContext, job, maxResults, startIndex, false);
                    }
                };
                // execute the task, no need to wait for result
                Future<DataMatrix> future = ExecutionManager.INSTANCE.submit(userContext.getCustomerId(), task);
                if (style == Style.HTML) {
                    // in that case we want to wait for the result in order to get data info
                    try {
                        DataMatrix matrix = future.get();
                        int end = startIndex + maxResults;
                        if (end > matrix.getRows().size())
                            end = matrix.getRows().size();
                        info = getAnalyticsResultInfo(end - startIndex, startIndex, matrix);
                    } catch (ExecutionException e) {
                        throwCauseException(e);
                    }
                }
            }
            specs.data = new Data();
            if (!outputConfig.isHasMetricSeries()) {
                specs.data.url = buildAnalyticsQueryURI(userContext, query, "RECORDS", "DATA",
                        null/*default style*/, null).toString();
            } else {
                specs.data.url = buildAnalyticsQueryURI(userContext, query, "TRANSPOSE", "DATA",
                        null/*default style*/, null).toString();
            }
            specs.data.format = new Format();
            specs.data.format.type = FormatType.json;// lowercase only!
        } else {
            throw new APIException("undefined value for data parameter, must be EMBEDED or URL");
        }
        // mark
        if (outputConfig.isTimeseries()) {
            specs.mark = Mark.line;
        } else {
            specs.mark = Mark.bar;
        }
        // size
        if (specs.encoding.row == null && specs.encoding.column == null) {
            specs.config.cell = new VegaliteSpecs.Cell(640, 400);
        }
        //
        ViewReply reply = new ViewReply();
        reply.setQuery(query);
        reply.setResult(specs);
        //
        if (envelope == null) {
            envelope = computeEnvelope(query);
        }
        //
        if (style != null && style == Style.HTML) {
            return createHTMLPageView(space, view, info, reply);
        } else if (envelope == null || envelope.equals("") || envelope.equalsIgnoreCase("RESULT")) {
            return Response.ok(reply.getResult(), MediaType.APPLICATION_JSON_TYPE.toString()).build();
        } else if (envelope.equalsIgnoreCase("ALL")) {
            return Response.ok(reply, MediaType.APPLICATION_JSON_TYPE.toString()).build();
        } else {
            throw new InvalidIdAPIException("invalid parameter envelope=" + envelope + ", must be ALL, RESULT",
                    true);
        }
    }

    /**
     * @param matrix
     * @return
     */
    private Info getAnalyticsResultInfo(Integer pageSize, Integer startIndex, DataMatrix matrix) {
        AnalyticsResult.Info info = new Info();
        info.setFromCache(matrix.isFromCache());
        info.setFromSmartCache(matrix.isFromSmartCache());// actually we don't know the origin, see T1851
        info.setExecutionDate(matrix.getExecutionDate().toString());
        info.setStartIndex(startIndex);
        info.setPageSize(pageSize);
        info.setTotalSize(matrix.getRows().size());
        return info;
    }

    private String getBookmarkNavigationPath(Bookmark bookmark) {
        String path = bookmark.getPath();
        if (path.startsWith("/USER/")) {
            int pos = path.indexOf("/", 6);
            if (pos >= 0) {
                path = path.substring(pos);
            } else {
                path = "";// remove all, the path is in form /USERS/id
            }
            path = MYBOOKMARKS_FOLDER.getSelfRef() + "/" + path;
        }
        if (path.endsWith("/"))
            path = path.substring(0, path.length() - 1);
        return path;
    }

    private String getPageTitle(Space space) {
        if (space.hasBookmark()) {
            String path = getBookmarkNavigationPath(space.getBookmark());
            return path + "/" + space.getBookmark().getName();
        } else {
            return "/PROJECTS/" + space.getUniverse().getProject().getName() + "/" + space.getDomain().getName();
        }
    }

    private URI getParentLink(Space space) {
        if (space == null)
            return null;
        if (space.hasBookmark()) {
            String path = getBookmarkNavigationPath(space.getBookmark());
            return getPublicBaseUriBuilder().path("/analytics").queryParam(PARENT_PARAM, path)
                    .queryParam(STYLE_PARAM, "HTML").queryParam("access_token", userContext.getToken().getOid())
                    .build();
        } else {
            return getPublicBaseUriBuilder().path("/analytics")
                    .queryParam(PARENT_PARAM, "/PROJECTS/" + space.getUniverse().getProject().getName())
                    .queryParam(STYLE_PARAM, "HTML").queryParam("access_token", userContext.getToken().getOid())
                    .build();
        }
    }

    private void createHTMLtitle(StringBuilder html, String title, String BBID, URI backLink) {
        html.append(
                "<div class=\"logo\"><span>Open Bouquet Analytics Rest <b style='color:#ee7914;'>API</b> Viewer / STYLE=HTML</span><hr/></div>");
        html.append("<h2>");
        if (title != null) {
            html.append(title);
        }
        if (BBID != null) {
            html.append("&nbsp;[ID=" + BBID + "]");
        }
        html.append("</h2>");
        if (backLink != null)
            html.append("<a href=\"" + backLink
                    + "\"><img src='\n'"
                    + ">back to parent</a>");
        html.append("</div><hr>");
    }

    private StringBuilder createHTMLHeader(String title) {
        StringBuilder html = new StringBuilder("<html><title>List: " + title + "</title>");
        html.append("<style>" + "* {font-family: \"Helvetica Neue\",Helvetica,Arial,sans-serif; color: #666; }"
                + "table.data {border-collapse: collapse;width: 100%;}"
                + "th, td {text-align: left;padding: 8px; vertical-align: top;}"
                + ".data tr:nth-child(even) {background-color: #f2f2f2}"
                + ".data th {background-color: #ee7914;color: white;}" + ".vega-actions a {margin-right:10px;}"
                + ".tooltip {\n" + "    position: relative;\n" + "    display: inline-block;\n"
                + "    border-bottom: 1px dotted black;\n" + "}\n" + ".tooltip .tooltiptext {\n"
                + "    visibility: hidden;\n" + "    width: 300px;\n" + "    background-color: black;\n"
                + "    color: #fff;\n" + "    text-align: center;\n" + "    border-radius: 6px;\n"
                + "    padding: 5px 0;\n" + "    position: absolute;\n" + "    z-index: 1;\n" + "    top: 150%;\n"
                + "    left: 50%;\n" + "    margin-left: -30px;\n" + "}\n" + ".tooltip .tooltiptext::after {\n"
                + "    content: \"\";\n" + "    position: absolute;\n" + "    bottom: 100%;\n" + "    left: 50%;\n"
                + "    margin-left: -5px;\n" + "    border-width: 5px;\n" + "    border-style: solid;\n"
                + "    border-color: transparent transparent black transparent;\n" + "}\n"
                + ".tooltip:hover .tooltiptext {\n" + "    visibility: visible;\n" + "}" + "hr {border: none; "
                + "color: Gainsboro ;\n" + "background-color: Gainsboro ;\n" + "height: 3px;}"
                + "input[type=date], input[type=text], select {\n" + "    padding: 4px 4px;\n"
                + "    margin: 4px 0;\n" + "    display: inline-block;\n" + "    border: 1px solid #ccc;\n"
                + "    border-radius: 4px;\n" + "    box-sizing: border-box;\n" + "}\n" + "input[type=submit] {\n"
                + "    font-size: 1.3em;" + "    width: 200px;\n" + "    background-color: #ee7914;\n"
                + "    color: white;\n" + "    padding: 14px 20px;\n" + "    margin: 0 auto;\n"
                + "    display: block;" + "    border: none;\n" + "    border-radius: 4px;\n"
                + "    cursor: pointer;\n" + "}\n" + "input[type=text].q {\n" + "    box-sizing: border-box;\n"
                + "    border: 2px solid #ccc;\n" + "    border-radius: 4px;\n" + "    font-size: 16px;\n"
                + "    background-color: white;\n"
                + "    background-image: url('');\n"
                + "    background-position: 10px 10px; \n" + "    background-repeat: no-repeat;\n"
                + "    padding: 12px 20px 12px 40px;" + "}" + "input[type=submit]:hover {\n"
                + "    background-color: #ab570e;\n" + "}\n" + "fieldset {\n" + "    margin-top: 40px;\n" + "}\n"
                + "legend {\n" + "    font-size: 1.3em;\n" + "}\n" + "table.controls {\n"
                + "    border-collapse:separate; \n" + "    border-spacing: 0 20px;\n" + "}\n" + "body {\n"
                + "    margin: 8px;\n" + "}\n" + ".footer a, .footer a:hover, .footer a:visited {\n"
                + "    color: White;\n" + "}\n" + ".footer {\n" + "    background: #5A5A5A;\n"
                + "    padding: 10px;\n" + "    margin: 20px -8px -8px -8px;\n" + "}\n" + ".footer p {\n"
                + "    text-align: center;\n" + "    color: White;\n" + "    width: 100%;\n" + "}\n" + ".header {\n"
                + "    margin: -8px 0 0 0;\n" + "}\n" + "h2 { margin-bottom:0px;}\n" + ".logo {\n"
                + "    margin: 0;\n" + "    height: 63px;\n"
                + "    background: url('');\n"
                + "    background-repeat: no-repeat;\n" + "}\n" + ".logo span {\n" + "    line-height: 63px;\n"
                + "    vertical-align: middle;\n" + "    color: #5e5e5e;\n" + "    padding-left: 60px;\n"
                + "    font-size: 20px;\n" + "}\n" + ".header hr {\n" + "    margin: 0 -8px 0 -8px;\n" + "}\n"
                + ".period input {\n" + "    margin-left: 10px;\n" + "    margin-right: 10px;\n" + "}\n"
                + ".instructions {\n" + "    font-size: 1.2em;\n" + "    font-style: italic;\n" + "}" + "</style>");
        // drag & drop support
        html.append("<script>\n" + "function allowDrop(ev) {\n" + "    ev.preventDefault();\n" + "}\n"
                + "function drag(ev, text) {\n" + "    ev.dataTransfer.setData(\"text\", \"'\"+text+\"'\");\n"
                + "}\n" + "function drop(ev) {\n" + "    ev.preventDefault();\n"
                + "    var data = ev.dataTransfer.getData(\"text\");\n" + "    ev.target.value = data;\n" + "}\n"
                + "</script>");
        html.append("<body>");
        return html;
    }

    private Response createHTMLPageList(AppContext ctx, NavigationQuery query, NavigationResult result) {
        String title = (query.getParent() != null && query.getParent().length() > 0) ? query.getParent() : "Root";
        StringBuilder html = createHTMLHeader("List: " + title);
        createHTMLtitle(html, title, null, result.getParent().getUpLink());
        // form
        html.append("<form><table>");
        html.append("<tr><td><input size=50 class='q' type='text' name='q' placeholder='filter the list' value='"
                + (query.getQ() != null ? query.getQ() : "") + "'></td>"
                + "<td><input type=\"submit\" value=\"Filter\"></td></tr>");
        html.append("<input type='hidden' name='parent' value='"
                + (query.getParent() != null ? query.getParent() : "") + "'>");
        if (query.getStyle() != null)
            html.append("<input type='hidden' name='style' value='"
                    + (query.getStyle() != null ? query.getStyle() : "") + "'>");
        if (query.getVisibility() != null)
            html.append("<input type='hidden' name='visibility' value='"
                    + (query.getVisibility() != null ? query.getVisibility() : "") + "'>");
        if (query.getHiearchy() != null)
            html.append("<input type='hidden' name='hierarchy' value='" + query.getHiearchy() + "'>");
        html.append("<input type='hidden' name='access_token' value='" + ctx.getToken().getOid() + "'>");
        html.append("</table></form>");
        //
        // parent description
        if (result.getParent() != null && result.getParent().getDescription() != null
                && result.getParent().getDescription().length() > 0) {
            html.append("<p><i>" + result.getParent().getDescription() + "</i></p>");
        }
        // coontent
        html.append("<table style='border-collapse:collapse'>");
        for (NavigationItem item : result.getChildren()) {
            html.append("<tr>");
            html.append("<td valign='top'>" + item.getType() + "</td>");
            if (item.getLink() != null) {
                html.append("<td valign='top'><a href=\"" + StringEscapeUtils.escapeHtml4(item.getLink().toString())
                        + "\">" + item.getName() + "</a>");
            } else {
                html.append("<td valign='top'>" + item.getName());
            }
            if (item.getObjectLink() != null) {
                html.append("&nbsp;[<a href=\"" + StringEscapeUtils.escapeHtml4(item.getObjectLink().toString())
                        + "\">info</a>]");
            }
            if (item.getViewLink() != null) {
                html.append("&nbsp;[<a href=\"" + StringEscapeUtils.escapeHtml4(item.getViewLink().toString())
                        + "\">view</a>]");
            }
            if (item.getDescription() != null && item.getDescription().length() > 0) {
                html.append("<br><i>" + (item.getDescription() != null ? item.getDescription() : "") + "</i>");
            }
            html.append("</td>");
            if (item.getAttributes() != null) {
                for (Entry<String, String> entry : item.getAttributes().entrySet()) {
                    html.append("<td valign='top'>" + entry.getKey() + "=" + entry.getValue() + "</td>");
                }
            }
        }
        html.append("</table>");
        createHTMLAPIpanel(html, "listContent");
        html.append("</body></html>");
        return Response.ok(html.toString(), "text/html").build();
    }

    private Response createHTMLPageView(Space space, ViewQuery view, Info info, ViewReply reply) {
        String title = getPageTitle(space);
        StringBuilder html = createHTMLHeader("View: " + title);
        if (getPublicBaseUriBuilder().build().getScheme().equalsIgnoreCase("https")) {
            html.append(
                    "<script src=\"https://d3js.org/d3.v3.min.js\" charset=\"utf-8\"></script>\r\n<script src=\"https://vega.github.io/vega/vega.js\" charset=\"utf-8\"></script>\r\n<script src=\"https://vega.github.io/vega-lite/vega-lite.js\" charset=\"utf-8\"></script>\r\n<script src=\"https://vega.github.io/vega-editor/vendor/vega-embed.js\" charset=\"utf-8\"></script>\r\n\r\n");
        } else {
            html.append(
                    "<script src=\"http://d3js.org/d3.v3.min.js\" charset=\"utf-8\"></script>\r\n<script src=\"http://vega.github.io/vega/vega.js\" charset=\"utf-8\"></script>\r\n<script src=\"http://vega.github.io/vega-lite/vega-lite.js\" charset=\"utf-8\"></script>\r\n<script src=\"http://vega.github.io/vega-editor/vendor/vega-embed.js\" charset=\"utf-8\"></script>\r\n\r\n");
        }
        html.append("<body>");
        createHTMLtitle(html, title, view.getBBID(), getParentLink(space));
        createHTMLproblems(html, reply.getQuery().getProblems());
        html.append(
                "<div id=\"vis\"></div>\r\n\r\n<script>\r\nvar embedSpec = {\r\n  mode: \"vega-lite\", renderer:\"svg\",  spec:");
        html.append(writeVegalightSpecs(reply.getResult()));
        Encoding channels = reply.getResult().encoding;
        html.append(
                "}\r\nvg.embed(\"#vis\", embedSpec, function(error, result) {\r\n  // Callback receiving the View instance and parsed Vega spec\r\n  // result.view is the View, which resides under the '#vis' element\r\n});\r\n</script>\r\n");
        createHTMLpagination(html, view, info);
        // data-link
        URI dataLink = buildAnalyticsQueryURI(userContext, reply.getQuery(), "RECORDS", "ALL", Style.HTML, null);
        html.append("<p><a href=\"" + StringEscapeUtils.escapeHtml4(dataLink.toASCIIString())
                + "\">view query data</a></p>");
        //
        html.append("<form>");
        createHTMLfilters(html, reply.getQuery());
        html.append("<table>" + "<tr><td>x</td><td><input type=\"text\" size=30 name=\"x\" value=\""
                + getFieldValue(view.getX()) + "\"></td><td>"
                + (channels.x != null ? "as <b>" + channels.x.field + "</b>" : "") + "</td></tr>"
                + "<tr><td>y</td><td><input type=\"text\" size=30 name=\"y\" value=\"" + getFieldValue(view.getY())
                + "\"></td><td>" + (channels.y != null ? "as <b>" + channels.y.field + "</b>" : "") + "</td></tr>"
                + "<tr><td>color</td><td><input type=\"text\" size=30 name=\"color\" value=\""
                + getFieldValue(view.getColor()) + "\"></td><td>"
                + (channels.color != null ? "as <b>" + channels.color.field + "</b>" : "") + "</td></tr>"
                + "<tr><td>size</td><td><input type=\"text\" size=30 name=\"size\" value=\""
                + getFieldValue(view.getSize()) + "\"></td><td>"
                + (channels.size != null ? "as <b>" + channels.size.field + "</b>" : "") + "</td></tr>"
                + "<tr><td>column</td><td><input type=\"text\" size=30 name=\"column\" value=\""
                + getFieldValue(view.getColumn()) + "\"></td><td>"
                + (channels.column != null ? "as <b>" + channels.column.field + "</b>" : "") + "</td></tr>"
                + "<tr><td>row</td><td><input type=\"text\" size=30 name=\"row\" value=\""
                + getFieldValue(view.getRow()) + "\"></td><td>"
                + (channels.row != null ? "as <b>" + channels.row.field + "</b>" : "") + "</td></tr>");
        // metrics -- display the actual metrics
        html.append("<tr><td valign='top'>metrics</td><td>");
        createHTMLinputArray(html, "text", "metrics", reply.getQuery().getMetrics());
        html.append(
                "</td><td>Use the metrics parameters if you want to view multiple metrics on the same graph. Then you can use the <b>__VALUE</b> expression in channel to reference the metrics' value, and the <b>__METRICS</b> to get the metrics' name as a series.<br>If you need only a single metrics, you can directly define it in a channel, e.g. <code>y=count()</code>.");
        html.append("</td></tr>");
        // limits, maxResults, startIndex
        html.append("<tr><td>limit</td><td>");
        html.append("<input type=\"text\" name=\"limit\" value=\"" + getFieldValue(view.getLimit(), -1) + "\">");
        html.append("</td></tr>");
        html.append("<tr><td>maxResults</td><td>");
        html.append("<input type=\"text\" name=\"maxResults\" value=\"" + getFieldValue(view.getMaxResults(), -1)
                + "\">");
        html.append("</td></tr>");
        html.append("<tr><td>startIndex</td><td>");
        html.append("<input type=\"text\" name=\"startIndex\" value=\"" + getFieldValue(view.getStartIndex(), 0)
                + "\"></td><td><i>index is zero-based, so use the #count of the last row to view the next page</i>");
        html.append("</td></tr>");
        html.append("</table>" + "<input type=\"hidden\" name=\"style\" value=\"HTML\">"
                + "<input type=\"hidden\" name=\"access_token\" value=\""
                + space.getUniverse().getContext().getToken().getOid() + "\">"
                + "<input type=\"submit\" value=\"Refresh\">" + "</form>");
        createHTMLscope(html, space, reply.getQuery());
        createHTMLAPIpanel(html, "viewAnalysis");
        html.append("</body>\r\n</html>");
        return Response.ok(html.toString(), "text/html; charset=UTF-8").build();
    }

    /**
     * @param result
     * @return
     */
    private String writeVegalightSpecs(VegaliteSpecs specs) {
        ObjectMapper mapper = new ObjectMapper();
        try {
            mapper.setSerializationInclusion(Include.NON_NULL);
            return mapper.writeValueAsString(specs);
        } catch (JsonProcessingException e) {
            throw new APIException("failed to write vegalite specs to JSON", e, true);
        }
    }

    /**
     * displays query problems
     * @param html
     * @param query
     */
    private void createHTMLproblems(StringBuilder html, List<Problem> problems) {
        if (problems != null && problems.size() > 0) {
            html.append(
                    "<div class='problems' style='border:1px solid red; background-color:lightpink;'>There are some problems with the query:");
            for (Problem problem : problems) {
                html.append("<li>" + problem.getSeverity().toString() + ": " + problem.getSubject() + ": "
                        + problem.getMessage() + "</li>");
            }
            html.append("</div>");
        }
    }

    private SimpleDateFormat htmlDateFormat = new SimpleDateFormat("yyyy-MM-dd");

    private String formatDateForWeb(String jsonFormat) {
        try {
            Date date = ServiceUtils.getInstance().toDate(jsonFormat);
            return htmlDateFormat.format(date);
        } catch (ParseException e) {
            return jsonFormat;
        }
    }

    private String getDate(List<String> dates, int pos) {
        if (dates != null && !dates.isEmpty() && pos < dates.size()) {
            return formatDateForWeb(getFieldValue(dates.get(pos)));
        } else {
            return "";
        }
    }

    /**
     * create a filter HTML snippet
     * @param query
     * @return
     */
    private void createHTMLfilters(StringBuilder html, AnalyticsQuery query) {
        html.append("<table><tr><td>");
        // period
        html.append(
                "<span class='tooltip'>period: <span class='tooltiptext'>the period defines a dimension or expression of a type date that is used to filter the query or view. You can use the __PERIOD expression as a alias to it.</span></span>");
        html.append("</td><td>");
        html.append("<input type='text' size=30 name='period' value='" + getFieldValue(query.getPeriod()) + "'>");
        // timeframe
        html.append(
                "&nbsp;<span class='tooltip'>timeframe <span class='tooltiptext'>the timeframe defines the period range to filter. You can use an array of two dates for lower/upper bounds (inclusive). Or some alias like __ALL, __LAST_DAY, __LAST_7_DAYS, __CURRENT_MONTH, __PREVIOUS_MONTH, __CURRENT_YEAR, __PREVIOOUS_YEAR</span></span>");
        html.append("&nbsp;from:&nbsp;<input type='text' name='timeframe' value='"
                + getDate(query.getTimeframe(), 0) + "'>");
        html.append("&nbsp;to:&nbsp;<input type='text' name='timeframe' value='" + getDate(query.getTimeframe(), 1)
                + "'>");
        // compare
        html.append(
                "&nbsp;<span class='tooltip'>compareTo <span class='tooltiptext'>Activate and define the compare to period. You can use an array of two dates for lower/upper bounds (inclusive). Or some alias like __COMPARE_TO_PREVIOUS_PERIOD, __COMPARE_TO_PREVIOUS_MONTH, __COMPARE_TO_PREVIOOUS_YEAR</span></span>");
        html.append("&nbsp;from:&nbsp;<input type='text' name='compareTo' value='"
                + getDate(query.getCompareTo(), 0) + "'>");
        html.append("&nbsp;to:&nbsp;<input type='text' name='compareTo' value='" + getDate(query.getCompareTo(), 1)
                + "'>");
        html.append("</td></tr>");
        // filters
        html.append("<tr><td>");
        html.append(
                "<span class='tooltip'>filters:<span class='tooltiptext'>Define the filters to apply to results. A filter must be a valid conditional expression. If no filter is defined, the subject default config will apply. You can use the * token to extend the subject default configuration.</span></span>&nbsp;");
        html.append("</td><td>");
        if (query.getFilters() != null && query.getFilters().size() > 0) {
            for (String filter : query.getFilters()) {
                html.append(
                        "<input type='text' size=50 name='filters' value='" + getFieldValue(filter) + "'>&nbsp;");
            }
        }
        html.append("<input type='text' size=50 name='filters' value='' placeholder='type formula'>");
        html.append("</td></tr></table>");
    }

    private static final String axis_style = "display: inline-block;border:1px solid;border-radius:5px;background-color:LavenderBlush ;margin:1px;";
    private static final String metric_style = "display: inline-block;border:1px solid;border-radius:5px;background-color:Lavender;margin:1px;";
    private static final String func_style = "display: inline-block;border:1px solid;border-radius:5px;background-color:ghostwhite;margin:1px;";
    private static final String other_style = "display: inline-block;border:1px solid;border-radius:5px;background-color:azure;margin:1px;";

    private void createHTMLscope(StringBuilder html, Space space, AnalyticsQuery query) {
        html.append(
                "<fieldset><legend>Query scope: <i>this is the list of objects you can combine to build expressions in the query</i></legend>");
        html.append("<table>");
        html.append("<tr><td></td><td>You can Drag & Drop expression into input fields</td></tr>");
        html.append("<tr><td>GroupBy:</td><td>");
        for (Axis axis : space.A(true)) {// only print the visible scope
            try {
                IDomain image = axis.getDefinitionSafe().getImageDomain();
                if (!image.isInstanceOf(IDomain.OBJECT)) {
                    DimensionIndex index = axis.getIndex();
                    html.append("<span draggable='true' style='" + axis_style + "'");
                    ExpressionAST expr = axis.getDefinitionSafe();
                    html.append("title='" + getExpressionValueType(expr).toString() + ": ");
                    if (axis.getDescription() != null) {
                        html.append(axis.getDescription());
                    }
                    if (index.getErrorMessage() != null) {
                        html.append("\nError:" + index.getErrorMessage());
                    }
                    html.append("'");
                    html.append(" ondragstart='drag(event,\"" + index.getDimensionName() + "\")'>");
                    if (index.getErrorMessage() == null) {
                        html.append("&nbsp;" + index.getDimensionName() + "&nbsp;");
                    } else {
                        html.append("&nbsp;<del>" + index.getDimensionName() + "</del>&nbsp;");
                    }
                    html.append("</span>");
                }
            } catch (Exception e) {
                // ignore
            }
        }
        html.append("</td></tr>");
        html.append("<tr><td>Metrics:</td><td>");
        for (Measure m : space.M()) {
            if (m.getMetric() != null && !m.getMetric().isDynamic()) {
                html.append("<span draggable='true'  style='" + metric_style + "'");
                ExpressionAST expr = m.getDefinitionSafe();
                html.append("title='" + getExpressionValueType(expr).toString() + ": ");
                if (m.getDescription() != null) {
                    html.append(m.getDescription());
                }
                html.append("'");
                html.append(" ondragstart='drag(event,\"" + m.getName() + "\")'");
                html.append(">&nbsp;" + m.getName() + "&nbsp;</span>");
            }
        }
        html.append("</td></tr></table>");
        URI scopeLink = getPublicBaseUriBuilder().path("/analytics/{reference}/scope")
                .queryParam("style", Style.HTML).queryParam("access_token", userContext.getToken().getOid())
                .build(query.getBBID());
        html.append(
                "<a href=\"" + StringEscapeUtils.escapeHtml4(scopeLink.toASCIIString()) + "\">View the scope</a>");
        html.append("</fieldset>");
    }

    /**
     * @param space
     * @param suggestions
     * @param values 
     * @param types 
     * @param expression 
     * @return
     */
    private Response createHTMLPageScope(Space space, ExpressionSuggestion suggestions, String BBID, String value,
            ObjectType[] types, ValueType[] values) {
        String title = getPageTitle(space);
        StringBuilder html = createHTMLHeader("Scope: " + title);
        createHTMLtitle(html, title, BBID, getParentLink(space));
        if (value != null && value.length() > 0 && suggestions.getValidateMessage() != null
                && suggestions.getValidateMessage().length() > 0) {
            createHTMLproblems(html, Collections
                    .singletonList(new Problem(Severity.WARNING, value, suggestions.getValidateMessage())));
        }
        html.append("<form>");
        html.append("<p>Expression:<input type='text' name='value' size=100 value='" + getFieldValue(value)
                + "' placeholder='type expression to validate it or to filter the suggestion list'></p>");
        html.append("<fieldset><legend>Filter by expression type</legend>");
        html.append(
                "<input type='checkbox' name='types' value='" + ObjectType.DIMENSION + "'>" + ObjectType.DIMENSION);
        html.append("<input type='checkbox' name='types' value='" + ObjectType.COLUMN + "'>" + ObjectType.COLUMN);
        html.append(
                "<input type='checkbox' name='types' value='" + ObjectType.RELATION + "'>" + ObjectType.RELATION);
        html.append("<input type='checkbox' name='types' value='" + ObjectType.METRIC + "'>" + ObjectType.METRIC);
        html.append(
                "<input type='checkbox' name='types' value='" + ObjectType.FUNCTION + "'>" + ObjectType.FUNCTION);
        html.append("</fieldset>");
        html.append("<fieldset><legend>Filter by expression value</legend>");
        html.append("<input type='checkbox' name='values' value='" + ValueType.DATE + "'>" + ValueType.DATE);
        html.append("<input type='checkbox' name='values' value='" + ValueType.STRING + "'>" + ValueType.STRING);
        html.append(
                "<input type='checkbox' name='values' value='" + ValueType.CONDITION + "'>" + ValueType.CONDITION);
        html.append("<input type='checkbox' name='values' value='" + ValueType.NUMERIC + "'>" + ValueType.NUMERIC);
        html.append(
                "<input type='checkbox' name='values' value='" + ValueType.AGGREGATE + "'>" + ValueType.AGGREGATE);
        html.append("</fieldset>");
        html.append("<input type=\"hidden\" name=\"style\" value=\"HTML\">"
                + "<input type=\"hidden\" name=\"access_token\" value=\""
                + space.getUniverse().getContext().getToken().getOid() + "\">"
                + "<input type=\"submit\" value=\"Refresh\">");
        html.append("</form>");
        html.append(
                "<p><i> This is the list of all available expressions and function in this scope. Relation expression can be composed in order to navigate the data model.</i></p>");
        html.append("<table>");
        for (ExpressionSuggestionItem item : suggestions.getSuggestions()) {
            html.append("<tr><td>");
            html.append(item.getObjectType() + "</td><td>");
            html.append(item.getValueType() + "</td><td>");
            String style = other_style;
            if (item.getObjectType() == ObjectType.DIMENSION)
                style = axis_style;
            if (item.getObjectType() == ObjectType.METRIC)
                style = metric_style;
            if (item.getObjectType() == ObjectType.FUNCTION)
                style = func_style;
            html.append("<span style='" + style + "'>&nbsp;" + item.getDisplay() + "&nbsp;</span>");
            if (item.getSuggestion() != null) {
                URI link = getPublicBaseUriBuilder().path("/analytics/{reference}/scope")
                        .queryParam("value", value + item.getSuggestion()).queryParam("style", Style.HTML)
                        .queryParam("access_token", userContext.getToken().getOid()).build(BBID);
                html.append(
                        "&nbsp;[<a href=\"" + StringEscapeUtils.escapeHtml4(link.toASCIIString()) + "\">+</a>]");
            }
            if (item.getExpression() != null && item.getExpression() instanceof AxisExpression) {
                AxisExpression ref = (AxisExpression) item.getExpression();
                Axis axis = ref.getAxis();
                if (axis.getDimensionType() == Type.CATEGORICAL) {
                    URI link = getPublicBaseUriBuilder().path("/analytics/{reference}/facets/{facetId}")
                            .queryParam("style", Style.HTML)
                            .queryParam("access_token", userContext.getToken().getOid())
                            .build(BBID, item.getSuggestion());
                    html.append("&nbsp;[<a href=\"" + StringEscapeUtils.escapeHtml4(link.toASCIIString())
                            + "\">Indexed</a>]");
                } else if (axis.getDimensionType() == Type.CONTINUOUS) {
                    URI link = getPublicBaseUriBuilder().path("/analytics/{reference}/facets/{facetId}")
                            .queryParam("style", Style.HTML)
                            .queryParam("access_token", userContext.getToken().getOid())
                            .build(BBID, item.getSuggestion());
                    html.append("&nbsp;[<a href=\"" + StringEscapeUtils.escapeHtml4(link.toASCIIString())
                            + "\">Period</a>]");
                }
            }
            if (item.getDescription() != null && item.getDescription().length() > 0) {
                html.append("<br><i>" + item.getDescription() + "</i>");
            }
            html.append("</td></tr>");
        }
        html.append("</table>");
        createHTMLAPIpanel(html, "scopeAnalysis");
        html.append("</body></html>");
        return Response.ok(html.toString(), "text/html").build();
    }

    private void createHTMLAPIpanel(StringBuilder html, String method) {
        html.append("<fieldset><legend>API reference</legend>");
        // compute the raw URI
        UriBuilder builder = getPublicBaseUriBuilder().path(uriInfo.getPath());
        MultivaluedMap<String, String> parameters = uriInfo.getQueryParameters();
        parameters.remove(ACCESS_TOKEN_PARAM);
        parameters.remove(STYLE_PARAM);
        parameters.remove(ENVELOPE_PARAM);
        for (Entry<String, List<String>> parameter : parameters.entrySet()) {
            for (String value : parameter.getValue()) {
                builder.queryParam(parameter.getKey(), value);
            }
        }
        html.append(
                "<p>Request URL: <i>this URL will require authentication</i></p><div style='display:block;'><pre style='background-color: #fcf6db;border: 1px solid #e5e0c6; width:1024px; max-height: 400px;overflow-y: auto;'>"
                        + StringEscapeUtils.escapeHtml4(builder.build().toString()) + "</pre></div>");
        String curlURL = "\"" + (StringEscapeUtils.escapeHtml4(builder.build().toString()).replace("'", "'"))
                + "\"";
        html.append(
                "<p>CURL: <i>the command is authorized with the current token</i></p><div style='display:block;'><pre style='background-color: #fcf6db;border: 1px solid #e5e0c6; width:1024px; max-height: 400px;overflow-y: auto;'>curl -X GET --header 'Accept: application/json' --header 'Authorization: Bearer "
                        + userContext.getToken().getOid() + "' " + curlURL + "</pre></div>");
        createHTMLswaggerLink(html, method);
        html.append("</fieldset>");
        html.append(
                "<div class=\"footer\"><p>Powered by <a href=\"http://openbouquet.io/\">Open Bouquet</a> <i style='color:white;'>the Analytics Rest API</i></p></div>\n");
    }

    private void createHTMLswaggerLink(StringBuilder html, String method) {
        String baseUrl = "";
        try {
            baseUrl = "?url="
                    + URLEncoder.encode(getPublicBaseUriBuilder().path("swagger.json").build().toString(), "UTF-8")
                    + "";
        } catch (UnsupportedEncodingException | IllegalArgumentException | UriBuilderException e) {
            // default
        }
        html.append(
                "<p>the OB Analytics API provides more parameters... check in <a target='swagger' href='http://swagger.squidsolutions.com/"
                        + baseUrl + "#!/analytics/" + method + "'>swagger UI</a> for details</p>");
    }

    private ValueType getExpressionValueType(ExpressionAST expr) {
        IDomain image = expr.getImageDomain();
        return ExpressionSuggestionHandler.computeValueTypeFromImage(image);
    }

    private String getFieldValue(Object var) {
        if (var == null)
            return "";
        else
            return var.toString().replaceAll("\"", "&quot;").replaceAll("'", "&#x27;");
    }

    private String getFieldValue(Object var, Object defaultValue) {
        if (var == null)
            return defaultValue.toString();
        else
            return var.toString().replaceAll("\"", "&quot;").replaceAll("'", "&#x27;");
    }

    /**
     * @param uriInfo
     * @param userContext 
     * @param localScope 
     * @param BBID 
     * @param query
     * @return
     * @throws ScopeException 
     */
    protected URI buildExportURI(AppContext userContext, String BBID, AnalyticsQuery query, String filename)
            throws ScopeException {
        UriBuilder builder = getPublicBaseUriBuilder()
                .path("/analytics/{" + BBID_PARAM_NAME + "}/export/{filename}");
        addAnalyticsQueryParams(builder, query, null, null);
        builder.queryParam("access_token", userContext.getToken().getOid());
        return builder.build(BBID, filename);
    }

    private URI buildAnalyticsViewURI(AppContext userContext, ViewQuery query, String data, String envelope,
            Style style, HashMap<String, Object> override) {
        UriBuilder builder = getPublicBaseUriBuilder().path("/analytics/{" + BBID_PARAM_NAME + "}/view");
        addAnalyticsQueryParams(builder, query, style, override);
        if (query.getX() != null)
            builder.queryParam(VIEW_X_PARAM, query.getX());
        if (query.getY() != null)
            builder.queryParam(VIEW_Y_PARAM, query.getY());
        if (query.getColor() != null)
            builder.queryParam(VIEW_COLOR_PARAM, query.getColor());
        if (query.getSize() != null)
            builder.queryParam(VIEW_SIZE_PARAM, query.getSize());
        if (query.getColumn() != null)
            builder.queryParam(VIEW_COLUMN_PARAM, query.getColumn());
        if (query.getRow() != null)
            builder.queryParam(VIEW_ROW_PARAM, query.getRow());
        if (data != null)
            builder.queryParam(DATA_PARAM, data);
        if (envelope != null)
            builder.queryParam(ENVELOPE_PARAM, envelope);
        builder.queryParam("access_token", userContext.getToken().getOid());
        return builder.build(query.getBBID());
    }

    private URI buildAnalyticsQueryURI(AppContext userContext, AnalyticsQuery query, String data, String envelope,
            Style style, HashMap<String, Object> override) {
        UriBuilder builder = getPublicBaseUriBuilder().path("/analytics/{" + BBID_PARAM_NAME + "}/query");
        addAnalyticsQueryParams(builder, query, style, override);
        if (data != null)
            builder.queryParam(DATA_PARAM, data);
        if (envelope != null)
            builder.queryParam(ENVELOPE_PARAM, envelope);
        builder.queryParam("access_token", userContext.getToken().getOid());
        return builder.build(query.getBBID());
    }

    private URI buildAnalyticsExportURI(AppContext userContext, AnalyticsQuery query, String filename) {
        UriBuilder builder = getPublicBaseUriBuilder()
                .path("/analytics/{" + BBID_PARAM_NAME + "}/export/{filename}");
        addAnalyticsQueryParams(builder, query, null, null);
        builder.queryParam("access_token", userContext.getToken().getOid());
        return builder.build(query.getBBID(), filename);
    }

    /**
     * @param override 
     * @throws ScopeException 
     * 
     */
    private void addAnalyticsQueryParams(UriBuilder builder, AnalyticsQuery query, Style style,
            HashMap<String, Object> override) {
        if (query.getGroupBy() != null) {
            for (String item : query.getGroupBy()) {
                builder.queryParam(GROUP_BY_PARAM, item);
            }
        }
        if (query.getMetrics() != null) {
            for (String item : query.getMetrics()) {
                builder.queryParam(METRICS_PARAM, item);
            }
        }
        if (query.getFilters() != null) {
            for (String item : query.getFilters()) {
                builder.queryParam(FILTERS_PARAM, item);
            }
        }
        if (query.getPeriod() != null)
            builder.queryParam(PERIOD_PARAM, query.getPeriod());
        if (query.getTimeframe() != null) {
            for (String item : query.getTimeframe()) {
                builder.queryParam(TIMEFRAME_PARAM, item);
            }
        }
        if (query.getCompareTo() != null) {
            for (String item : query.getCompareTo()) {
                builder.queryParam(COMPARETO_PARAM, item);
            }
        }
        if (query.getOrderBy() != null) {
            for (String item : query.getOrderBy()) {
                builder.queryParam(ORDERBY_PARAM, item);
            }
        }
        if (query.getRollups() != null)
            builder.queryParam(ROLLUP_PARAM, query.getRollups());
        // limit override
        if (override != null && override.containsKey(LIMIT_PARAM)) {
            if (override.get(LIMIT_PARAM) != null)
                builder.queryParam(LIMIT_PARAM, override.get(LIMIT_PARAM));
        } else if (query.getLimit() != null)
            builder.queryParam(LIMIT_PARAM, query.getLimit());
        if (query.getBeyondLimit() != null) {
            for (String value : query.getBeyondLimit()) {
                builder.queryParam("beyondLimit", value);
            }
        }
        // maxResults override
        if (override != null && override.containsKey(MAX_RESULTS_PARAM)) {
            if (override.get(MAX_RESULTS_PARAM) != null)
                builder.queryParam(MAX_RESULTS_PARAM, override.get(MAX_RESULTS_PARAM));
        } else if (query.getMaxResults() != null)
            builder.queryParam(MAX_RESULTS_PARAM, query.getMaxResults());
        // startIndex override
        if (override != null && override.containsKey(START_INDEX_PARAM)) {
            if (override.get(START_INDEX_PARAM) != null)
                builder.queryParam(START_INDEX_PARAM, override.get(START_INDEX_PARAM));
        } else if (query.getStartIndex() != null)
            builder.queryParam(START_INDEX_PARAM, query.getStartIndex());
        //
        if (query.getLazy() != null)
            builder.queryParam(LAZY_PARAM, query.getLazy());
        if (style != null) {
            builder.queryParam(STYLE_PARAM, style);
        } else if (query.getStyle() != null)
            builder.queryParam(STYLE_PARAM, query.getStyle());
    }

    private Data transformToVegaData(AnalyticsQuery query, DataMatrix matrix, String format) {
        IDataMatrixConverter<Object[]> converter = getConverter(format);
        Data data = new Data();
        data.values = converter.convert(query, matrix);
        return data;
    }

    private IDataMatrixConverter<Object[]> getConverter(String format) {
        if (format.equalsIgnoreCase("RECORDS")) {
            return new RecordConverter();
        } else if (format.equalsIgnoreCase("TRANSPOSE")) {
            return new TransposeConverter();
        } else {
            throw new InvalidIdAPIException("invalid format=" + format, true);
        }
    }

    /**
     * moved some legacy code out of AnalysisJobComputer
     * => still need to bypass the ProjectAnalysisJob
     * @param ctx
     * @param job
     * @param maxResults
     * @param startIndex
     * @param lazy
     * @return
     * @throws ComputingException
     * @throws InterruptedException
     */
    private DataMatrix compute(AppContext ctx, ProjectAnalysisJob job, Integer maxResults, Integer startIndex,
            boolean lazy) throws ComputingException, InterruptedException {
        // build the analysis
        long start = System.currentTimeMillis();
        logger.info("Starting preview compute for job " + job.getId());
        DashboardAnalysis analysis;
        try {
            analysis = AnalysisJobComputer.buildDashboardAnalysis(ctx, job, lazy);
        } catch (Exception e) {
            throw new ComputingException(e);
        }
        // run the analysis
        DataMatrix datamatrix = ComputingService.INSTANCE.glitterAnalysis(analysis, null);
        if (lazy && (datamatrix == null)) {
            throw new NotInCacheException("Lazy preview, analysis " + analysis.getJobId() + "  not in cache");
        } else {
            job.setRedisKey(datamatrix.getRedisKey());
            long stop = System.currentTimeMillis();
            logger.info("task=" + this.getClass().getName() + " method=compute" + " jobid="
                    + job.getId().getAnalysisJobId() + " duration=" + (stop - start));
            JobStats queryLog = new JobStats(job.getId().getAnalysisJobId(), "AnalysisJobComputer.compute",
                    (stop - start), job.getId().getProjectId());
            queryLog.setError(false);
            PerfDB.INSTANCE.save(queryLog);
            return datamatrix;
        }
    }

    // Execution management

    /**
     * list the execution status for a given analysis. Note: an analysis can spam multiple queries.
     * @param request
     * @param key
     * @param style 
     * @return
     */
    public List<QueryWorkerJobStatus> getStatus(AppContext userContext, String key) {
        // first check if the query is available
        String customerId = userContext.getCustomerId();
        List<QueryWorkerJobStatus> queries = RedisCacheManager.getInstance().getQueryServer()
                .getOngoingQueries(customerId);
        queries.addAll(DomainHierarchyManager.INSTANCE.getOngoingQueries(customerId));
        List<QueryWorkerJobStatus> results = new ArrayList<>();
        for (QueryWorkerJobStatus query : queries) {
            if (query.getJobID().equals(key)) {
                ProjectPK projectPK = query.getProjectPK();
                try {
                    Project project = ProjectManager.INSTANCE.getProject(userContext, projectPK);
                    // restrict to privileged user
                    if (checkACL(userContext, project, query)) {
                        results.add(query);
                    }
                } catch (ScopeException e) {
                    // ignore
                }
            }
        }
        //
        return results;
    }

    /**
     * @param userContext
     * @param key
     * @return
     */
    public boolean cancelQuery(AppContext userContext, String key) {
        List<QueryWorkerJobStatus> jobs = getStatus(userContext, key);
        boolean result = true;
        for (QueryWorkerJobStatus job : jobs) {
            if (!RedisCacheManager.getInstance().getQueryServer().cancelOngoingQuery(userContext.getCustomerId(),
                    job.getKey())) {
                result = false;
            }
        }
        return result;
    }

    private boolean checkACL(AppContext userContext, Project project, QueryWorkerJobStatus query) {
        if (AccessRightsUtils.getInstance().hasRole(userContext, project, Role.WRITE)) {
            return true;
        } else if (AccessRightsUtils.getInstance().hasRole(userContext, project, Role.READ)) {
            // or to the query owner
            if (query.getUserID().equals(userContext.getUser().getOid())) {
                return true;
            }
        }
        // else
        return false;
    }

}