org.apache.camel.component.google.bigquery.GoogleBigQueryProducer Maven / Gradle / Ivy
/*
* 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.google.bigquery;
import java.util.ArrayList;
import java.util.List;
import java.util.Map;
import com.google.api.client.util.Strings;
import com.google.cloud.bigquery.BigQuery;
import com.google.cloud.bigquery.InsertAllRequest;
import com.google.cloud.bigquery.InsertAllResponse;
import org.apache.camel.Exchange;
import org.apache.camel.ExchangePropertyKey;
import org.apache.camel.support.DefaultProducer;
import org.apache.camel.util.ObjectHelper;
import org.slf4j.Logger;
import org.slf4j.LoggerFactory;
/**
* Generic BigQuery Producer
*/
public class GoogleBigQueryProducer extends DefaultProducer {
private static final Logger LOG = LoggerFactory.getLogger(GoogleBigQueryProducer.class);
private final GoogleBigQueryConfiguration configuration;
private BigQuery bigquery;
public GoogleBigQueryProducer(BigQuery bigquery, GoogleBigQueryEndpoint endpoint,
GoogleBigQueryConfiguration configuration) {
super(endpoint);
this.bigquery = bigquery;
this.configuration = configuration;
}
/**
* The method converts a single incoming message into a List
*/
private static List prepareExchangeList(Exchange exchange) {
List entryList;
if (null == exchange.getProperty(ExchangePropertyKey.GROUPED_EXCHANGE)) {
entryList = new ArrayList<>();
entryList.add(exchange);
} else {
entryList = (List) exchange.getProperty(ExchangePropertyKey.GROUPED_EXCHANGE);
}
return entryList;
}
/**
* Process the exchange
*
* The incoming exchange can be a grouped exchange in which case all the exchanges will be combined.
*
* The incoming can be
*
* - A map where all map keys will map to field records. One map object maps to one bigquery row
* - A list of maps. Each entry in the list will map to one bigquery row
*
* The incoming message is expected to be a List of Maps The assumptions: - All incoming records go into the same
* table - Incoming records sorted by the timestamp
*/
@Override
public void process(Exchange exchange) throws Exception {
List exchanges = prepareExchangeList(exchange);
List processGroup = new ArrayList<>();
String partitionDecorator = "";
String suffix = "";
String tableId = configuration.getTableId() == null ? "" : configuration.getTableId();
int totalProcessed = 0;
for (Exchange ex : exchanges) {
String tmpPartitionDecorator = ex.getIn().getHeader(GoogleBigQueryConstants.PARTITION_DECORATOR, "", String.class);
String tmpSuffix = ex.getIn().getHeader(GoogleBigQueryConstants.TABLE_SUFFIX, "", String.class);
String tmpTableId = ex.getIn().getHeader(GoogleBigQueryConstants.TABLE_ID, tableId, String.class);
if (tmpTableId.isEmpty()) {
throw new IllegalArgumentException(
"tableId need to be specified in one of endpoint configuration or exchange header");
}
// Ensure all rows of same request goes to same table and suffix
if (!tmpPartitionDecorator.equals(partitionDecorator) || !tmpSuffix.equals(suffix) || !tmpTableId.equals(tableId)) {
if (!processGroup.isEmpty()) {
totalProcessed += process(tableId, partitionDecorator, suffix, processGroup, exchange.getExchangeId());
}
processGroup.clear();
partitionDecorator = tmpPartitionDecorator;
suffix = tmpSuffix;
tableId = tmpTableId;
}
processGroup.add(ex);
}
if (!processGroup.isEmpty()) {
totalProcessed += process(tableId, partitionDecorator, suffix, processGroup, exchange.getExchangeId());
}
if (totalProcessed == 0) {
LOG.debug("The incoming message is either null or empty for exchange {}", exchange.getExchangeId());
}
}
private int process(String tableId, String partitionDecorator, String suffix, List exchanges, String exchangeId)
throws Exception {
String tableIdWithPartition = Strings.isNullOrEmpty(partitionDecorator)
? tableId
: (tableId + "$" + partitionDecorator);
List apiRequestRows = new ArrayList<>();
for (Exchange ex : exchanges) {
Object entryObject = ex.getIn().getBody();
if (entryObject instanceof List) {
for (Map entry : (List
© 2015 - 2025 Weber Informatics LLC | Privacy Policy