/*
* Copyright 2014 - 2017 Blazebit.
*
* 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 com.blazebit.persistence.impl.hibernate;
import org.hibernate.MappingException;
import org.hibernate.cache.CacheException;
import org.hibernate.cache.spi.access.CollectionRegionAccessStrategy;
import org.hibernate.cfg.Configuration;
import org.hibernate.engine.spi.SessionFactoryImplementor;
import org.hibernate.engine.spi.SessionImplementor;
import org.hibernate.engine.spi.SubselectFetch;
import org.hibernate.loader.collection.CollectionInitializer;
import org.hibernate.mapping.Collection;
import org.hibernate.persister.collection.BasicCollectionPersister;
/**
* @author Jan-Willem Gmelig Meyling
* @since 1.2.0
*/
public class CustomBasicCollectionPersister extends BasicCollectionPersister {
public CustomBasicCollectionPersister(Collection collection, CollectionRegionAccessStrategy cacheAccessStrategy, Configuration cfg, SessionFactoryImplementor factory) throws MappingException, CacheException {
super(collection, cacheAccessStrategy, cfg, factory);
}
@Override
protected CollectionInitializer createSubselectInitializer(SubselectFetch subselect, SessionImplementor session) {
// Hibernate before 4 couldn't find the correct from clause
StringBuilder sb = null;
String subselectQuery = subselect.toSubselectString( getCollectionType().getLHSPropertyName() );
int parens = 0;
for (int i = 0; i < subselectQuery.length(); i++) {
final char c = subselectQuery.charAt(i);
if (c == '(') {
parens++;
} else if (c == ')') {
parens--;
if (parens < 0) {
// This is the case when we have a CTE
int fromIndex = subselectQuery.indexOf("from");
int otherFromIndex = subselectQuery.indexOf("from", i);
sb = new StringBuilder(fromIndex + (subselectQuery.length() - otherFromIndex));
sb.append(subselectQuery, 0, fromIndex);
sb.append(subselectQuery, otherFromIndex, subselectQuery.length());
break;
}
}
}
if (sb != null) {
subselectQuery = sb.toString();
}
return new CustomSubselectCollectionLoader(
this,
subselectQuery,
subselect.getResult(),
subselect.getQueryParameters(),
subselect.getNamedParameterLocMap(),
session.getFactory(),
session.getLoadQueryInfluencers()
);
}
}