/* * Copyright 2012 JBoss Inc * * Licensed 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.artificer.client; import org.artificer.client.i18n.Messages; import org.artificer.client.query.QueryResultSet; import org.artificer.common.error.ArtificerServerException; import org.joda.time.DateTime; import org.joda.time.format.ISODateTimeFormat; import java.util.ArrayList; import java.util.Calendar; import java.util.Date; import java.util.HashSet; import java.util.List; import java.util.Set; /** * An S-RAMP Query, created by the {@link ArtificerAtomApiClient} from an xpath template. The * xpath template is of the same form as a typical JDBC statement (using the ? character * for replacements). Following are same example usages: * * @author eric.wittmann@redhat.com */ public class ArtificerClientQuery { private ArtificerAtomApiClient client; private String queryTemplate; private List<String> replacementParams = new ArrayList<String>(); private int startIndex = 0; private int count = 20; private String orderBy = "name"; private boolean ascending = true; private Set<String> propertyNames = new HashSet<String>(); /** * Constructor. * @param client * @param queryTemplate */ protected ArtificerClientQuery(ArtificerAtomApiClient client, String queryTemplate) { this.client = client; this.queryTemplate = queryTemplate; } /** * Sets a parameter on the query - this should match up to a ? in the * query template provided. * @param param */ public ArtificerClientQuery parameter(String param) { replacementParams.add("'" + param.replace("'", "''") + "'"); return this; } /** * Sets a parameter on the query - this should match up to a ? in the * query template provided. * @param param */ public ArtificerClientQuery parameter(int param) { replacementParams.add(String.valueOf(param)); return this; } /** * Sets a parameter on the query - this should match up to a ? in the * query template provided. * @param param */ public ArtificerClientQuery parameter(long param) { replacementParams.add(String.valueOf(param)); return this; } /** * Sets a parameter on the query - this should match up to a ? in the * query template provided. * @param param */ public ArtificerClientQuery parameter(double param) { replacementParams.add(String.valueOf(param)); return this; } /** * Sets a parameter on the query - this should match up to a ? in the * query template provided. Note: this will add a date to the query. In * order to be more precise and send a full DateTime, use the Calendar * form of this method: <code>parameter(Calendar param)</code> * @param param */ public ArtificerClientQuery parameter(Date param) { String val = ISODateTimeFormat.date().print(new DateTime(param)); replacementParams.add("'" + val + "'"); return this; } /** * Sets a parameter on the query. This should match up to a ? in the * query template provided. note: this will add a DateTime to the query. * @param param */ public ArtificerClientQuery parameter(Calendar param) { String val = ISODateTimeFormat.dateTimeNoMillis().print(new DateTime(param)); replacementParams.add("'" + val + "'"); return this; } /** * Sets a parameter on the query - this should match up to a ? in the * query template provided. * @param param */ public ArtificerClientQuery parameter(float param) { replacementParams.add(String.valueOf(param)); return this; } /** * Sets a parameter on the query - this should match up to a ? in the * query template provided. * @param param */ public ArtificerClientQuery parameter(Number param) { replacementParams.add(param.toString()); return this; } /** * @param startIndex the startIndex to set */ public ArtificerClientQuery startIndex(int startIndex) { this.startIndex = startIndex; return this; } /** * @param count the count to set */ public ArtificerClientQuery count(int count) { this.count = count; return this; } /** * @param orderBy the orderBy to set */ public ArtificerClientQuery orderBy(String orderBy) { this.orderBy = orderBy; return this; } /** * Sets ascending to true. */ public ArtificerClientQuery ascending() { this.ascending = true; return this; } /** * Sets ascending to false. */ public ArtificerClientQuery descending() { this.ascending = false; return this; } /** * @param propertyName property name to include in the result */ public ArtificerClientQuery propertyName(String propertyName) { this.propertyNames.add(propertyName); return this; } /** * Formats the query given the replacement params, then issues the query * to the S-RAMP repository and returns the result. * @throws org.artificer.atom.err.ArtificerAtomException * @throws ArtificerClientException */ public QueryResultSet query() throws ArtificerClientException, ArtificerServerException { String query = formatQuery(); return client.query(query, startIndex, count, orderBy, ascending, propertyNames); } /** * Formats the query by replacing all ? characters with their replacement * parameters. * @throws ArtificerClientException */ private String formatQuery() throws ArtificerClientException { StringBuilder builder = new StringBuilder(); String [] xpathSegments = queryTemplate.split("\\?"); int paramCounter = 0; for (String segment : xpathSegments) { builder.append(segment); boolean isLastSegment = segment == xpathSegments[xpathSegments.length - 1]; if (!isLastSegment) { if (paramCounter >= replacementParams.size()) throw new ArtificerClientException(Messages.i18n.format("TOO_FEW_QUERY_PARAMS")); String param = replacementParams.get(paramCounter); builder.append(param); paramCounter++; } } if (replacementParams.size() > paramCounter) throw new ArtificerClientException(Messages.i18n.format("TOO_MANY_QUERY_PARAMS")); return builder.toString(); } }