org.apache.camel.component.salesforce.AnalyticsApiIntegrationTest.java Source code

Java tutorial

Introduction

Here is the source code for org.apache.camel.component.salesforce.AnalyticsApiIntegrationTest.java

Source

/**
 * Licensed to the Apache Software Foundation (ASF) under one or more
 * contributor license agreements.  See the NOTICE file distributed with
 * this work for additional information regarding copyright ownership.
 * The ASF licenses this file to You under the Apache License, Version 2.0
 * (the "License"); you may not use this file except in compliance with
 * the License.  You may obtain a copy of the License at
 *
 *      http://www.apache.org/licenses/LICENSE-2.0
 *
 * Unless required by applicable law or agreed to in writing, software
 * distributed under the License is distributed on an "AS IS" BASIS,
 * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
 * See the License for the specific language governing permissions and
 * limitations under the License.
 */
package org.apache.camel.component.salesforce;

import java.io.FileInputStream;
import java.util.Collection;
import java.util.HashMap;
import java.util.List;
import java.util.Map;
import java.util.Properties;
import java.util.TreeMap;
import java.util.concurrent.TimeUnit;

import org.apache.camel.builder.RouteBuilder;
import org.apache.camel.component.salesforce.api.SalesforceReportResultsToListConverter;
import org.apache.camel.component.salesforce.api.dto.analytics.reports.AsyncReportResults;
import org.apache.camel.component.salesforce.api.dto.analytics.reports.QueryRecordsReport;
import org.apache.camel.component.salesforce.api.dto.analytics.reports.Report;
import org.apache.camel.component.salesforce.api.dto.analytics.reports.ReportDescription;
import org.apache.camel.component.salesforce.api.dto.analytics.reports.ReportInstance;
import org.apache.camel.component.salesforce.api.dto.analytics.reports.ReportMetadata;
import org.apache.camel.component.salesforce.api.dto.analytics.reports.ReportStatusEnum;
import org.apache.camel.component.salesforce.api.dto.analytics.reports.SyncReportResults;
import org.apache.camel.dataformat.csv.CsvDataFormat;
import org.apache.commons.csv.CSVFormat;
import org.junit.BeforeClass;
import org.junit.Test;
import org.junit.experimental.theories.DataPoints;
import org.junit.experimental.theories.Theories;
import org.junit.experimental.theories.Theory;
import org.junit.runner.RunWith;
import org.slf4j.Logger;
import org.slf4j.LoggerFactory;

/**
 * Integration test for Salesforce analytics API endpoints. 
 */
@RunWith(Theories.class)
public class AnalyticsApiIntegrationTest extends AbstractSalesforceTestBase {

    private static final Logger LOG = LoggerFactory.getLogger(AnalyticsApiIntegrationTest.class);
    private static final int RETRY_DELAY = 5000;
    private static final int REPORT_RESULT_RETRIES = 5;
    private static final String[] REPORT_OPTIONS = new String[] {
            SalesforceReportResultsToListConverter.INCLUDE_HEADERS,
            SalesforceReportResultsToListConverter.INCLUDE_DETAILS,
            SalesforceReportResultsToListConverter.INCLUDE_SUMMARY };
    private static final int NUM_OPTIONS = REPORT_OPTIONS.length;
    private static final int[] POWERS = new int[] { 4, 2, 1 };

    private static String[] testReportNames;
    private boolean bodyMetadata;

    /**
     * Get test report developer names as data points.
     * @return test report developer names in test-salesforce-login.properties
     * @throws Exception
     */
    @DataPoints
    public static String[] getTestReportDeveloperNames() throws Exception {
        return testReportNames;
    }

    @BeforeClass
    public static void getReportNames() throws Exception {
        // get test report names
        Properties testProperties = new Properties();
        testProperties.load(new FileInputStream(LoginConfigHelper.TEST_LOGIN_PROPERTIES));

        Map<String, String> reports = new TreeMap<String, String>();
        for (Map.Entry<Object, Object> entry : testProperties.entrySet()) {
            final String key = entry.getKey().toString();
            if (key.matches("report.[0-9]+")) {
                reports.put(key, entry.getValue().toString());
            }
        }
        assertFalse("Missing entries report.[0-9]+=<Report DeveloperName> in "
                + LoginConfigHelper.TEST_LOGIN_PROPERTIES, reports.isEmpty());

        final Collection<String> reportNames = reports.values();
        testReportNames = reportNames.toArray(new String[reportNames.size()]);
    }

    @Test
    public void testGetRecentReports() throws Exception {

        final List recentReports = template().requestBody("direct:getRecentReports", null, List.class);

        assertNotNull("getRecentReports", recentReports);
        assertFalse("getRecentReports empty", recentReports.isEmpty());
        LOG.debug("getRecentReports: {}", recentReports);
    }

    @Theory
    public void testReport(String reportName) throws Exception {

        log.info("Testing report {}...", reportName);

        // get Report Id
        final QueryRecordsReport reports = template().requestBody("direct:queryReport",
                "SELECT Id FROM Report WHERE DeveloperName='" + reportName + "'", QueryRecordsReport.class);

        assertNotNull("query", reports);
        final List<Report> reportsRecords = reports.getRecords();
        assertFalse("Report not found", reportsRecords.isEmpty());
        final String testReportId = reportsRecords.get(0).getId();
        assertNotNull(testReportId);

        // 1. getReportDescription
        final ReportDescription reportDescription = template().requestBody("direct:getReportDescription",
                testReportId, ReportDescription.class);

        assertNotNull("getReportDescriptions", reportDescription);
        LOG.debug("getReportDescriptions: {}", reportDescription);
        final ReportMetadata testReportMetadata = reportDescription.getReportMetadata();

        // 2. executeSyncReport
        // execute with no metadata
        SyncReportResults reportResults = template().requestBodyAndHeader("direct:executeSyncReport", testReportId,
                SalesforceEndpointConfig.INCLUDE_DETAILS, Boolean.TRUE, SyncReportResults.class);

        assertNotNull("executeSyncReport", reportResults);
        LOG.debug("executeSyncReport: {}", reportResults);

        // execute with metadata
        final Map<String, Object> headers = new HashMap<String, Object>();
        headers.put(SalesforceEndpointConfig.INCLUDE_DETAILS, Boolean.FALSE);
        Object body;
        if (!bodyMetadata) {
            headers.put(SalesforceEndpointConfig.REPORT_METADATA, testReportMetadata);
            body = testReportId;
        } else {
            body = testReportMetadata;
        }
        reportResults = template().requestBodyAndHeaders("direct:executeSyncReport", body, headers,
                SyncReportResults.class);

        assertNotNull("executeSyncReport with metadata", reportResults);
        LOG.debug("executeSyncReport with metadata: {}", reportResults);

        // 3. executeAsyncReport
        // execute with no metadata
        ReportInstance reportInstance = template().requestBodyAndHeader("direct:executeAsyncReport", testReportId,
                SalesforceEndpointConfig.INCLUDE_DETAILS, true, ReportInstance.class);

        assertNotNull("executeAsyncReport", reportInstance);
        LOG.debug("executeAsyncReport: {}", reportInstance);

        // execute with metadata
        headers.clear();
        headers.put(SalesforceEndpointConfig.INCLUDE_DETAILS, "true");
        if (!bodyMetadata) {
            headers.put(SalesforceEndpointConfig.REPORT_METADATA, testReportMetadata);
            body = testReportId;
            bodyMetadata = true;
        } else {
            body = testReportMetadata;
            bodyMetadata = false;
        }
        reportInstance = template().requestBodyAndHeaders("direct:executeAsyncReport", body, headers,
                ReportInstance.class);

        assertNotNull("executeAsyncReport with metadata", reportInstance);
        LOG.debug("executeAsyncReport with metadata: {}", reportInstance);
        final String testReportInstanceId = reportInstance.getId();

        // 4. getReportInstances
        final List reportInstances = template().requestBody("direct:getReportInstances", testReportId, List.class);

        assertNotNull("getReportInstances", reportInstances);
        assertFalse("getReportInstances empty", reportInstances.isEmpty());
        LOG.debug("getReportInstances: {}", reportInstances);

        // 5. getReportResults
        // wait for the report to complete
        boolean done = false;
        int tries = 0;
        AsyncReportResults asyncReportResults = null;
        while (!done) {
            asyncReportResults = template().requestBodyAndHeader("direct:getReportResults", testReportId,
                    SalesforceEndpointConfig.INSTANCE_ID, testReportInstanceId, AsyncReportResults.class);
            done = asyncReportResults != null
                    && (asyncReportResults.getAttributes().getStatus() == ReportStatusEnum.Success
                            || asyncReportResults.getAttributes().getStatus() == ReportStatusEnum.Error);
            if (!done) {
                // avoid flooding calls
                Thread.sleep(RETRY_DELAY);
                if (++tries > REPORT_RESULT_RETRIES) {
                    final long retrySeconds = TimeUnit.SECONDS.convert(tries * RETRY_DELAY, TimeUnit.MILLISECONDS);
                    fail("Async report result not available in " + retrySeconds + " seconds");
                }
            }
        }

        assertNotNull("getReportResults", asyncReportResults);
        assertEquals("getReportResults status", ReportStatusEnum.Success,
                asyncReportResults.getAttributes().getStatus());
        LOG.debug("getReportResults: {}", asyncReportResults);

        // 6. SalesforceReportResultsConverter tests
        // defaults
        String convertResults = template.requestBody("direct:convertResults", asyncReportResults, String.class);
        assertNotNull("default convertResults", convertResults);
        LOG.debug("Default options", convertResults);
        LOG.debug("{}", convertResults);

        // permutations of include details, include headers, include summary
        final boolean[] values = new boolean[NUM_OPTIONS];
        final int nIterations = (int) Math.pow(2, NUM_OPTIONS);

        for (int i = 0; i < nIterations; i++) {

            // toggle options
            for (int j = 0; j < NUM_OPTIONS; j++) {
                if (i % POWERS[j] == 0) {
                    values[j] = !values[j];
                }
            }

            log.debug("Options {} = {}", REPORT_OPTIONS, values);
            headers.clear();
            for (int j = 0; j < REPORT_OPTIONS.length; j++) {
                headers.put(REPORT_OPTIONS[j], values[j]);
            }
            convertResults = template.requestBodyAndHeaders("direct:convertResults", asyncReportResults, headers,
                    String.class);

            assertNotNull("convertResults", convertResults);
            LOG.debug("{}", convertResults);
        }
    }

    @Override
    protected RouteBuilder doCreateRouteBuilder() throws Exception {
        return new RouteBuilder() {
            @Override
            public void configure() throws Exception {

                // get Report SObject by DeveloperName
                from("direct:queryReport")
                        .to("salesforce:query?sObjectClass=" + QueryRecordsReport.class.getName());

                from("direct:getRecentReports").to("salesforce:getRecentReports");

                from("direct:getReportDescription").to("salesforce:getReportDescription");

                from("direct:executeSyncReport").to("salesforce:executeSyncReport");

                from("direct:executeAsyncReport").to("salesforce:executeAsyncReport?includeDetails=true");

                from("direct:getReportInstances").to("salesforce:getReportInstances");

                from("direct:getReportResults").to("salesforce:getReportResults");

                CsvDataFormat csv = new CsvDataFormat(CSVFormat.EXCEL);

                // type converter test
                from("direct:convertResults").convertBodyTo(List.class).marshal(csv);
            }
        };
    }
}