All Downloads are FREE. Search and download functionalities are using the official Maven repository.

net.sf.jasperreports.jakarta.ejbql.JRJpaDataSource Maven / Gradle / Ivy

/*
 * JasperReports - Free Java Reporting Library.
 * Copyright (C) 2001 - 2023 Cloud Software Group, Inc. All rights reserved.
 * http://www.jaspersoft.com
 *
 * Unless you have purchased a commercial license agreement from Jaspersoft,
 * the following license terms apply:
 *
 * This program is part of JasperReports.
 *
 * JasperReports is free software: you can redistribute it and/or modify
 * it under the terms of the GNU Lesser General Public License as published by
 * the Free Software Foundation, either version 3 of the License, or
 * (at your option) any later version.
 *
 * JasperReports 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. See the
 * GNU Lesser General Public License for more details.
 *
 * You should have received a copy of the GNU Lesser General Public License
 * along with JasperReports. If not, see .
 */
package net.sf.jasperreports.jakarta.ejbql;

import java.util.HashMap;
import java.util.Iterator;
import java.util.List;
import java.util.Map;

import net.sf.jasperreports.engine.JRException;
import net.sf.jasperreports.engine.JRField;
import net.sf.jasperreports.engine.JRRuntimeException;
import net.sf.jasperreports.engine.data.JRAbstractBeanDataSource;

/**
 * Java Persistence API data source that uses jakarta.persistence.Query.getResultList().
 * 

* The query result can be paginated by not retrieving all the rows at once. *

* Fields are mapped to values in the result following these rules: *

    *
  • if the query returns a single object/bean (e.g. SELECT m FROM Movie m or * SELECT NEW MovieDescription(m.title, m.genre) FROM Movie m), then the fields are * mapped to bean property names.
  • *
  • if the query returns multiple objects per row (e.g. SELECT m.title, m.gender FROM Movie m), * the fields are mapped using the following syntax: COLUMN_index[.property], with the * indexes starting from 1. Example mappings: COLUMN_1, COLUMN_2, COLUMN_2.title, COLUMN_2.movie.title.
  • *
* * @author Marcel Overdijk ([email protected]) * @see net.sf.jasperreports.jakarta.ejbql.JRJpaQueryExecuter#PROPERTY_JPA_QUERY_PAGE_SIZE */ public class JRJpaDataSource extends JRAbstractBeanDataSource { public static final String EXCEPTION_MESSAGE_KEY_INDEX_OUT_OF_BOUNDS = "data.jpa.index.out.of.bounds"; private static final String MAPPING_INDEX_PREFIX = "COLUMN_"; private static final int MAPPING_INDEX_PREFIX_LENGTH = MAPPING_INDEX_PREFIX.length(); private static final String MAPPING_INDEX_PROPERTY_SEP = "."; private static final int MAPPING_INDEX_PROPERTY_SEP_LENGTH = MAPPING_INDEX_PROPERTY_SEP.length(); private final JRJpaQueryExecuter queryExecuter; private final int pageSize; private int pageCount; private boolean nextPage; private List returnValues; private Iterator iterator; protected Object currentRow; private Map fieldValueReaders; public JRJpaDataSource(JRJpaQueryExecuter queryExecuter, int pageSize) { super(true); this.queryExecuter = queryExecuter; this.pageSize = pageSize; fieldValueReaders = new HashMap<>(); pageCount = 0; fetchPage(); } protected void fetchPage() { if (pageSize <= 0) { returnValues = queryExecuter.getResultList(); nextPage = false; } else { returnValues = queryExecuter.getResultList(pageCount * pageSize, pageSize); nextPage = returnValues.size() == pageSize; } ++pageCount; initIterator(); } @Override public boolean next() { if (iterator == null) { return false; } boolean hasNext = iterator.hasNext(); if (!hasNext && nextPage) { fetchPage(); hasNext = iterator != null && iterator.hasNext(); } if (hasNext) { currentRow = iterator.next(); } return hasNext; } @Override public void moveFirst() { if (pageCount == 1) { initIterator(); } else { pageCount = 0; fetchPage(); } } private void initIterator() { iterator = returnValues == null ? null : returnValues.iterator(); } @Override public Object getFieldValue(JRField field) throws JRException { FieldValueReader reader = getFieldValueReader(field); return reader.getValue(); } protected FieldValueReader getFieldValueReader(JRField field) { FieldValueReader reader = fieldValueReaders.get(field.getName()); if (reader == null) { String mapping = getPropertyName(field); reader = createReader(mapping); fieldValueReaders.put(field.getName(), reader); } return reader; } private FieldValueReader createReader(String mapping) { FieldValueReader reader = null; if (mapping.startsWith(MAPPING_INDEX_PREFIX)) { int propertySepIdx = mapping.indexOf(MAPPING_INDEX_PROPERTY_SEP, MAPPING_INDEX_PREFIX_LENGTH + 1); if (propertySepIdx < 0) { String indexStr = mapping.substring(MAPPING_INDEX_PREFIX_LENGTH); try { int index = Integer.parseInt(indexStr); reader = new IndexReader(index - 1); } catch (NumberFormatException e) { //ignore } } else { String indexStr = mapping.substring(MAPPING_INDEX_PREFIX_LENGTH, propertySepIdx); try { int index = Integer.parseInt(indexStr); String property = mapping.substring(propertySepIdx + MAPPING_INDEX_PROPERTY_SEP_LENGTH); reader = new IndexPropertyReader(index - 1, property); } catch (NumberFormatException e) { //ignore } } } //default to a property mapping if (reader == null) { reader = new PropertyReader(mapping); } return reader; } protected static interface FieldValueReader { Object getValue() throws JRException; } protected class PropertyReader implements FieldValueReader { private final String property; public PropertyReader(String property) { this.property = property; } @Override public Object getValue() throws JRException { return getBeanProperty(currentRow, property); } } protected class IndexReader implements FieldValueReader { private final int position; public IndexReader(int position) { this.position = position; } @Override public Object getValue() throws JRException { Object[] values = (Object[]) currentRow; if (position < 0 || position >= values.length) { throw new JRRuntimeException( EXCEPTION_MESSAGE_KEY_INDEX_OUT_OF_BOUNDS, new Object[]{position, values.length}); } return values[position]; } } protected class IndexPropertyReader implements FieldValueReader { private final int position; private final String property; public IndexPropertyReader(int position, String property) { this.position = position; this.property = property; } @Override public Object getValue() throws JRException { Object[] values = (Object[]) currentRow; if (position < 0 || position >= values.length) { throw new JRRuntimeException( EXCEPTION_MESSAGE_KEY_INDEX_OUT_OF_BOUNDS, new Object[]{position, values.length}); } return getBeanProperty(values[position], property); } } }




© 2015 - 2024 Weber Informatics LLC | Privacy Policy