blob: 82750df0df95e45b5dbe6cb7d43c3de5e68f35cf [file] [log] [blame]
/*
* 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.openjpa.persistence.jdbc.maps.m2mmapex1;
import java.util.Collection;
import java.util.HashMap;
import java.util.List;
import java.util.Map;
import javax.persistence.EntityManager;
import javax.persistence.EntityManagerFactory;
import javax.persistence.EntityTransaction;
import javax.persistence.Query;
import org.apache.openjpa.kernel.QueryImpl;
import org.apache.openjpa.persistence.test.AllowFailure;
import org.apache.openjpa.persistence.test.SQLListenerTestCase;
import junit.framework.Assert;
public class TestMany2ManyMapEx1 extends SQLListenerTestCase {
public int numEmployees = 2;
public int numPhoneNumbersPerEmployee = 2;
public Map<Integer, Employee> empMap = new HashMap<>();
public Map<Integer, PhoneNumber> phoneMap =
new HashMap<>();
public int empId = 1;
public int phoneId = 1;
public int divId = 1;
public int deptId = 10;
public List rsAllPhones = null;
public List rsAllEmps = null;
public List rsAllDivisions = null;
@Override
public void setUp() {
super.setUp(CLEAR_TABLES,
Department.class,
Division.class,
Employee.class,
PhoneNumber.class);
createObj(emf);
rsAllPhones = getAll(PhoneNumber.class);
rsAllEmps = getAll(Employee.class);
rsAllDivisions = getAll(Division.class);
}
@AllowFailure
public void testQueryInMemoryQualifiedId() throws Exception {
queryQualifiedId(true);
}
public void testQueryQualifiedId() throws Exception {
queryQualifiedId(false);
}
public void setCandidate(Query q, Class clz)
throws Exception {
org.apache.openjpa.persistence.QueryImpl q1 =
(org.apache.openjpa.persistence.QueryImpl) q;
org.apache.openjpa.kernel.Query q2 = q1.getDelegate();
org.apache.openjpa.kernel.QueryImpl qi = (QueryImpl) q2;
if (clz == PhoneNumber.class)
qi.setCandidateCollection(rsAllPhones);
else if (clz == Employee.class)
qi.setCandidateCollection(rsAllEmps);
else if (clz == Division.class)
qi.setCandidateCollection(rsAllDivisions);
}
public void queryQualifiedId(boolean inMemory) throws Exception {
EntityManager em = emf.createEntityManager();
String query = "select KEY(e), p from PhoneNumber p, " +
" in (p.emps) e order by p.number, e.empId";
Query q = em.createQuery(query);
if (inMemory)
setCandidate(q, PhoneNumber.class);
List rs = q.getResultList();
Division d = (Division) ((Object[]) rs.get(0))[0];
PhoneNumber p = (PhoneNumber) ((Object[]) rs.get(0))[1];
query = "select KEY(p) from Employee e, " +
" in (e.phones) p order by p.number, e.empId";
q = em.createQuery(query);
if (inMemory)
setCandidate(q, Employee.class);
rs = q.getResultList();
Department d2 = (Department) rs.get(0);
em.clear();
query = "select ENTRY(e) from PhoneNumber p, " +
" in (p.emps) e order by p.number, e.empId";
q = em.createQuery(query);
if (inMemory)
setCandidate(q, PhoneNumber.class);
rs = q.getResultList();
Map.Entry me = (Map.Entry) rs.get(0);
Division other = (Division) me.getKey();
assertEquals(d.getId(), other.getId());
assertEquals(d.getName(), other.getName());
// test navigation thru KEY
em.clear();
query = "select KEY(e), KEY(e).name from PhoneNumber p, " +
" in (p.emps) e order by p.number, e.empId";
q = em.createQuery(query);
if (inMemory)
setCandidate(q, PhoneNumber.class);
rs = q.getResultList();
Division d0 = (Division) ((Object[]) rs.get(0))[0];
String name = (String)((Object[]) rs.get(0))[1];
assertEquals(d0.getName(), name);
em.clear();
query = "select KEY(p), KEY(p).name from Employee e, " +
" in (e.phones) p order by p.number, e.empId";
q = em.createQuery(query);
if (inMemory)
setCandidate(q, Employee.class);
rs = q.getResultList();
d2 = (Department) ((Object[]) rs.get(0))[0];
String dname = (String) ((Object[]) rs.get(0))[1];
assertEquals(d2.getName(), dname);
// test ORDER BY qualified path
em.clear();
query = "select KEY(p), KEY(p).name from Employee e, " +
" in (e.phones) p ORDER BY KEY(p).name DESC";
q = em.createQuery(query);
if (inMemory)
setCandidate(q, Employee.class);
rs = q.getResultList();
String name1 = (String) ((Object[]) rs.get(0))[1];
em.clear();
query = "select KEY(p), KEY(p).name as name from Employee e, " +
" in (e.phones) p ORDER BY name DESC";
q = em.createQuery(query);
if (inMemory)
setCandidate(q, Employee.class);
rs = q.getResultList();
String name2 = (String) ((Object[]) rs.get(0))[1];
assertEquals(name1, name2);
// test GROUP BY qualified path
query = "select count(KEY(p).name) from Employee e, " +
" in (e.phones) p GROUP BY KEY(p).name";
q = em.createQuery(query);
if (inMemory)
setCandidate(q, Employee.class);
rs = q.getResultList();
em.clear();
query = "select p.division, KEY(p), KEY(p).name from Employee e, " +
" in (e.phones) p ORDER BY KEY(p).name DESC";
q = em.createQuery(query);
if (inMemory)
setCandidate(q, Employee.class);
rs = q.getResultList();
query = "select KEY(e) from PhoneNumber p, " +
" in (p.emps) e";
q = em.createQuery(query);
if (inMemory)
setCandidate(q, PhoneNumber.class);
rs = q.getResultList();
query = "select KEY(e) from PhoneNumber p " +
" left join p.emps e";
q = em.createQuery(query);
if (inMemory)
setCandidate(q, PhoneNumber.class);
rs = q.getResultList();
query = "select p.division, KEY(e), KEY(e).name as nm" +
" from PhoneNumber p, " +
" in (p.emps) e order by nm";
q = em.createQuery(query);
if (inMemory)
setCandidate(q, PhoneNumber.class);
rs = q.getResultList();
String n1 = ((Division) ((Object[]) rs.get(0))[1]).getName();
String n2 = (String) ((Object[]) rs.get(0))[2];
assertEquals(n1, n2);
query = "select d.name, KEY(e), KEY(e).name from PhoneNumber p, " +
" in (p.emps) e, Division d";
q = em.createQuery(query);
if (inMemory)
setCandidate(q, PhoneNumber.class);
rs = q.getResultList();
query = "select d.name, KEY(e), KEY(e).name from " +
"Division d join d.phone p, " +
" in (p.emps) e order by d.name";
q = em.createQuery(query);
if (inMemory)
setCandidate(q, Division.class);
rs = q.getResultList();
n1 = ((Division) ((Object[]) rs.get(0))[1]).getName();
n2 = (String) ((Object[]) rs.get(0))[2];
assertEquals(n1, n2);
em.close();
}
public void testQueryObject() throws Exception {
queryObj(emf);
findObj(emf);
}
public void createObj(EntityManagerFactory emf) {
EntityManager em = emf.createEntityManager();
EntityTransaction tran = em.getTransaction();
for (int i = 0; i < numEmployees; i++) {
Employee e = createEmployee(em, empId++);
empMap.put(e.getEmpId(), e);
}
tran.begin();
em.flush();
tran.commit();
em.close();
}
public Employee createEmployee(EntityManager em, int id) {
Employee e = new Employee();
e.setEmpId(id);
for (int i = 0; i < numPhoneNumbersPerEmployee; i++) {
PhoneNumber phoneNumber = new PhoneNumber();
phoneNumber.setNumber(phoneId++);
Division div = createDivision(em, divId++);
Department dept = createDepartment(em, deptId++);
phoneNumber.addEmployees(div, e);
e.addPhoneNumber(dept, phoneNumber);
phoneMap.put(phoneNumber.getNumber(), phoneNumber);
div.setPhone(phoneNumber);
phoneNumber.setDivision(div);
em.persist(phoneNumber);
em.persist(dept);
em.persist(div);
}
em.persist(e);
return e;
}
public Division createDivision(EntityManager em, int id) {
Division d = new Division();
d.setId(id);
d.setName("d" + id);
return d;
}
public Department createDepartment(EntityManager em, int id) {
Department d = new Department();
d.setId(id);
d.setName("dept" + id);
return d;
}
public void findObj(EntityManagerFactory emf) throws Exception {
EntityManager em = emf.createEntityManager();
Employee e = em.find(Employee.class, 1);
assertEmployee(e);
PhoneNumber p = em.find(PhoneNumber.class, 1);
assertPhoneNumber(p);
em.close();
}
public void queryObj(EntityManagerFactory emf) throws Exception {
queryEmployee(emf);
queryPhoneNumber(emf);
}
public void queryPhoneNumber(EntityManagerFactory emf) throws Exception {
EntityManager em = emf.createEntityManager();
EntityTransaction tran = em.getTransaction();
tran.begin();
Query q = em.createQuery("select p from PhoneNumber p");
List<PhoneNumber> ps = q.getResultList();
for (PhoneNumber p : ps) {
assertPhoneNumber(p);
}
tran.commit();
em.close();
}
public void queryEmployee(EntityManagerFactory emf) throws Exception {
EntityManager em = emf.createEntityManager();
EntityTransaction tran = em.getTransaction();
tran.begin();
Query q = em.createQuery("select e from Employee e");
List<Employee> es = q.getResultList();
for (Employee e : es) {
assertEmployee(e);
}
tran.commit();
em.close();
}
public void assertEmployee(Employee e) throws Exception {
int id = e.getEmpId();
Employee e0 = empMap.get(id);
Map<Department, PhoneNumber> phones0 = e0.getPhoneNumbers();
Map<Department, PhoneNumber> phones = e.getPhoneNumbers();
Assert.assertEquals(phones0.size(), phones.size());
checkPhoneMap(phones0, phones);
}
public void assertPhoneNumber(PhoneNumber p) throws Exception {
int number = p.getNumber();
PhoneNumber p0 = phoneMap.get(number);
Map<Division, Employee> es0 = p0.getEmployees();
Map<Division, Employee> es = p.getEmployees();
Assert.assertEquals(es0.size(), es.size());
checkEmpMap(es0, es);
}
public void checkPhoneMap(Map<Department, PhoneNumber> es0,
Map<Department, PhoneNumber> es) throws Exception {
Collection<Map.Entry<Department, PhoneNumber>> entrySets0 =
es0.entrySet();
for (Map.Entry<Department, PhoneNumber> entry0 : entrySets0) {
Department d0 = entry0.getKey();
PhoneNumber p0 = entry0.getValue();
PhoneNumber p = es.get(d0);
if (!p0.equals(p))
throw new Exception("Assertion failure");
}
}
public void checkEmpMap(Map<Division, Employee> es0,
Map<Division, Employee> es) throws Exception {
Collection<Map.Entry<Division, Employee>> entrySets0 = es0.entrySet();
for (Map.Entry<Division, Employee> entry0 : entrySets0) {
Division d0 = entry0.getKey();
Employee e0 = entry0.getValue();
Employee e = es.get(d0);
if (!e0.equals(e))
throw new Exception("Assertion failure");
}
}
}