mirror of
https://github.com/google/nomulus.git
synced 2025-07-12 14:08:18 +02:00
Add MapUserType to support converstion between Map and hstore (#443)
This commit is contained in:
parent
cf0e8e1b26
commit
6aa7c19344
8 changed files with 293 additions and 40 deletions
|
@ -14,20 +14,17 @@
|
||||||
|
|
||||||
package google.registry.persistence;
|
package google.registry.persistence;
|
||||||
|
|
||||||
import java.io.Serializable;
|
|
||||||
import java.sql.Array;
|
import java.sql.Array;
|
||||||
import java.sql.PreparedStatement;
|
import java.sql.PreparedStatement;
|
||||||
import java.sql.ResultSet;
|
import java.sql.ResultSet;
|
||||||
import java.sql.SQLException;
|
import java.sql.SQLException;
|
||||||
import java.sql.Types;
|
import java.sql.Types;
|
||||||
import java.util.Collection;
|
import java.util.Collection;
|
||||||
import java.util.Objects;
|
|
||||||
import org.hibernate.HibernateException;
|
import org.hibernate.HibernateException;
|
||||||
import org.hibernate.engine.spi.SharedSessionContractImplementor;
|
import org.hibernate.engine.spi.SharedSessionContractImplementor;
|
||||||
import org.hibernate.usertype.UserType;
|
|
||||||
|
|
||||||
/** Generic Hibernate user type to store/retrieve Java collection as an array in Cloud SQL. */
|
/** Generic Hibernate user type to store/retrieve Java collection as an array in Cloud SQL. */
|
||||||
public abstract class GenericCollectionUserType<T extends Collection> implements UserType {
|
public abstract class GenericCollectionUserType<T extends Collection> extends MutableUserType {
|
||||||
|
|
||||||
abstract T getNewCollection();
|
abstract T getNewCollection();
|
||||||
|
|
||||||
|
@ -62,16 +59,6 @@ public abstract class GenericCollectionUserType<T extends Collection> implements
|
||||||
return new int[] {getColumnType().getTypeCode()};
|
return new int[] {getColumnType().getTypeCode()};
|
||||||
}
|
}
|
||||||
|
|
||||||
@Override
|
|
||||||
public boolean equals(Object x, Object y) throws HibernateException {
|
|
||||||
return Objects.equals(x, y);
|
|
||||||
}
|
|
||||||
|
|
||||||
@Override
|
|
||||||
public int hashCode(Object x) throws HibernateException {
|
|
||||||
return x == null ? 0 : x.hashCode();
|
|
||||||
}
|
|
||||||
|
|
||||||
@Override
|
@Override
|
||||||
public Object nullSafeGet(
|
public Object nullSafeGet(
|
||||||
ResultSet rs, String[] names, SharedSessionContractImplementor session, Object owner)
|
ResultSet rs, String[] names, SharedSessionContractImplementor session, Object owner)
|
||||||
|
@ -98,30 +85,4 @@ public abstract class GenericCollectionUserType<T extends Collection> implements
|
||||||
Array arr = st.getConnection().createArrayOf(getColumnType().getTypeName(), list.toArray());
|
Array arr = st.getConnection().createArrayOf(getColumnType().getTypeName(), list.toArray());
|
||||||
st.setArray(index, arr);
|
st.setArray(index, arr);
|
||||||
}
|
}
|
||||||
|
|
||||||
// TODO(b/147489651): Investigate how to properly implement the below methods.
|
|
||||||
@Override
|
|
||||||
public Object deepCopy(Object value) throws HibernateException {
|
|
||||||
return value;
|
|
||||||
}
|
|
||||||
|
|
||||||
@Override
|
|
||||||
public boolean isMutable() {
|
|
||||||
return false;
|
|
||||||
}
|
|
||||||
|
|
||||||
@Override
|
|
||||||
public Serializable disassemble(Object value) throws HibernateException {
|
|
||||||
return (Serializable) value;
|
|
||||||
}
|
|
||||||
|
|
||||||
@Override
|
|
||||||
public Object assemble(Serializable cached, Object owner) throws HibernateException {
|
|
||||||
return cached;
|
|
||||||
}
|
|
||||||
|
|
||||||
@Override
|
|
||||||
public Object replace(Object original, Object target, Object owner) throws HibernateException {
|
|
||||||
return original;
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
|
|
@ -0,0 +1,57 @@
|
||||||
|
// Copyright 2020 The Nomulus Authors. All Rights Reserved.
|
||||||
|
//
|
||||||
|
// 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 google.registry.persistence;
|
||||||
|
|
||||||
|
import java.sql.PreparedStatement;
|
||||||
|
import java.sql.ResultSet;
|
||||||
|
import java.sql.SQLException;
|
||||||
|
import java.sql.Types;
|
||||||
|
import java.util.Map;
|
||||||
|
import org.hibernate.HibernateException;
|
||||||
|
import org.hibernate.engine.spi.SharedSessionContractImplementor;
|
||||||
|
import org.hibernate.usertype.UserType;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* A custom {@link UserType} used to convert a Java {@link Map<String, String>} to/from PostgreSQL
|
||||||
|
* hstore type. Per this <a href="https://www.postgresql.org/docs/current/hstore.html">doc</a>, as
|
||||||
|
* hstore keys and values are simply text strings, the type of key and value in the Java map has to
|
||||||
|
* be {@link String} as well.
|
||||||
|
*/
|
||||||
|
public class MapUserType extends MutableUserType {
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public int[] sqlTypes() {
|
||||||
|
return new int[] {Types.OTHER};
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public Class returnedClass() {
|
||||||
|
return Map.class;
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public Object nullSafeGet(
|
||||||
|
ResultSet rs, String[] names, SharedSessionContractImplementor session, Object owner)
|
||||||
|
throws HibernateException, SQLException {
|
||||||
|
return rs.getObject(names[0]);
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public void nullSafeSet(
|
||||||
|
PreparedStatement st, Object value, int index, SharedSessionContractImplementor session)
|
||||||
|
throws HibernateException, SQLException {
|
||||||
|
st.setObject(index, value);
|
||||||
|
}
|
||||||
|
}
|
|
@ -0,0 +1,63 @@
|
||||||
|
// Copyright 2020 The Nomulus Authors. All Rights Reserved.
|
||||||
|
//
|
||||||
|
// 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 google.registry.persistence;
|
||||||
|
|
||||||
|
import java.io.Serializable;
|
||||||
|
import java.util.Objects;
|
||||||
|
import org.hibernate.HibernateException;
|
||||||
|
import org.hibernate.usertype.UserType;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* An abstract class represents a mutable Hibernate user type which implements related methods
|
||||||
|
* defined in {@link UserType}.
|
||||||
|
*/
|
||||||
|
public abstract class MutableUserType implements UserType {
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public boolean equals(Object x, Object y) throws HibernateException {
|
||||||
|
return Objects.equals(x, y);
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public int hashCode(Object x) throws HibernateException {
|
||||||
|
return x == null ? 0 : x.hashCode();
|
||||||
|
}
|
||||||
|
|
||||||
|
// TODO(b/147489651): Investigate how to properly implement the below methods.
|
||||||
|
@Override
|
||||||
|
public Object deepCopy(Object value) throws HibernateException {
|
||||||
|
return value;
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public boolean isMutable() {
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public Serializable disassemble(Object value) throws HibernateException {
|
||||||
|
return (Serializable) value;
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public Object assemble(Serializable cached, Object owner) throws HibernateException {
|
||||||
|
return cached;
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public Object replace(Object original, Object target, Object owner) throws HibernateException {
|
||||||
|
return original;
|
||||||
|
}
|
||||||
|
}
|
|
@ -24,6 +24,7 @@ public class NomulusPostgreSQLDialect extends PostgreSQL95Dialect {
|
||||||
registerColumnType(Types.VARCHAR, "text");
|
registerColumnType(Types.VARCHAR, "text");
|
||||||
registerColumnType(Types.TIMESTAMP_WITH_TIMEZONE, "timestamptz");
|
registerColumnType(Types.TIMESTAMP_WITH_TIMEZONE, "timestamptz");
|
||||||
registerColumnType(Types.TIMESTAMP, "timestamptz");
|
registerColumnType(Types.TIMESTAMP, "timestamptz");
|
||||||
|
registerColumnType(Types.OTHER, "hstore");
|
||||||
for (ArrayColumnType arrayType : ArrayColumnType.values()) {
|
for (ArrayColumnType arrayType : ArrayColumnType.values()) {
|
||||||
registerColumnType(arrayType.getTypeCode(), arrayType.getTypeDdlName());
|
registerColumnType(arrayType.getTypeCode(), arrayType.getTypeDdlName());
|
||||||
}
|
}
|
||||||
|
|
|
@ -0,0 +1,140 @@
|
||||||
|
// Copyright 2020 The Nomulus Authors. All Rights Reserved.
|
||||||
|
//
|
||||||
|
// 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 google.registry.persistence;
|
||||||
|
|
||||||
|
import static com.google.common.truth.Truth.assertThat;
|
||||||
|
import static google.registry.model.transaction.TransactionManagerFactory.jpaTm;
|
||||||
|
import static org.junit.Assert.assertThrows;
|
||||||
|
|
||||||
|
import com.google.common.collect.ImmutableMap;
|
||||||
|
import google.registry.model.ImmutableObject;
|
||||||
|
import google.registry.model.transaction.JpaTestRules;
|
||||||
|
import google.registry.model.transaction.JpaTestRules.JpaIntegrationTestRule;
|
||||||
|
import java.util.Map;
|
||||||
|
import javax.persistence.Entity;
|
||||||
|
import javax.persistence.Id;
|
||||||
|
import javax.persistence.NoResultException;
|
||||||
|
import org.hibernate.annotations.Type;
|
||||||
|
import org.junit.Rule;
|
||||||
|
import org.junit.Test;
|
||||||
|
import org.junit.runner.RunWith;
|
||||||
|
import org.junit.runners.JUnit4;
|
||||||
|
|
||||||
|
/** Unit tests for {@link MapUserType}. */
|
||||||
|
@RunWith(JUnit4.class)
|
||||||
|
public class MapUserTypeTest {
|
||||||
|
|
||||||
|
// Note that JpaIntegrationTestRule is used here as the hstore extension is installed
|
||||||
|
// when nomulus.golden.sql is executed as the init script.
|
||||||
|
@Rule
|
||||||
|
public final JpaIntegrationTestRule jpaRule =
|
||||||
|
new JpaTestRules.Builder().withEntityClass(TestEntity.class).buildIntegrationTestRule();
|
||||||
|
|
||||||
|
@Test
|
||||||
|
public void roundTripConversion_returnsSameMap() {
|
||||||
|
Map<String, String> map = ImmutableMap.of("key1", "value1", "key2", "value2");
|
||||||
|
TestEntity testEntity = new TestEntity(map);
|
||||||
|
jpaTm().transact(() -> jpaTm().getEntityManager().persist(testEntity));
|
||||||
|
TestEntity persisted =
|
||||||
|
jpaTm().transact(() -> jpaTm().getEntityManager().find(TestEntity.class, "id"));
|
||||||
|
assertThat(persisted.map).containsExactly("key1", "value1", "key2", "value2");
|
||||||
|
}
|
||||||
|
|
||||||
|
@Test
|
||||||
|
public void testMerge_succeeds() {
|
||||||
|
Map<String, String> map = ImmutableMap.of("key1", "value1", "key2", "value2");
|
||||||
|
TestEntity testEntity = new TestEntity(map);
|
||||||
|
jpaTm().transact(() -> jpaTm().getEntityManager().persist(testEntity));
|
||||||
|
TestEntity persisted =
|
||||||
|
jpaTm().transact(() -> jpaTm().getEntityManager().find(TestEntity.class, "id"));
|
||||||
|
persisted.map = ImmutableMap.of("key3", "value3");
|
||||||
|
jpaTm().transact(() -> jpaTm().getEntityManager().merge(persisted));
|
||||||
|
TestEntity updated =
|
||||||
|
jpaTm().transact(() -> jpaTm().getEntityManager().find(TestEntity.class, "id"));
|
||||||
|
assertThat(updated.map).containsExactly("key3", "value3");
|
||||||
|
}
|
||||||
|
|
||||||
|
@Test
|
||||||
|
public void testNullValue_writesAndReadsNullSuccessfully() {
|
||||||
|
TestEntity testEntity = new TestEntity(null);
|
||||||
|
jpaTm().transact(() -> jpaTm().getEntityManager().persist(testEntity));
|
||||||
|
TestEntity persisted =
|
||||||
|
jpaTm().transact(() -> jpaTm().getEntityManager().find(TestEntity.class, "id"));
|
||||||
|
assertThat(persisted.map).isNull();
|
||||||
|
}
|
||||||
|
|
||||||
|
@Test
|
||||||
|
public void testEmptyCollection_writesAndReadsEmptyCollectionSuccessfully() {
|
||||||
|
TestEntity testEntity = new TestEntity(ImmutableMap.of());
|
||||||
|
jpaTm().transact(() -> jpaTm().getEntityManager().persist(testEntity));
|
||||||
|
TestEntity persisted =
|
||||||
|
jpaTm().transact(() -> jpaTm().getEntityManager().find(TestEntity.class, "id"));
|
||||||
|
assertThat(persisted.map).isEmpty();
|
||||||
|
}
|
||||||
|
|
||||||
|
@Test
|
||||||
|
public void testNativeQuery_succeeds() throws Exception {
|
||||||
|
executeNativeQuery(
|
||||||
|
"INSERT INTO \"TestEntity\" (name, map) VALUES ('id', 'key1=>value1, key2=>value2')");
|
||||||
|
|
||||||
|
assertThat(
|
||||||
|
getSingleResultFromNativeQuery(
|
||||||
|
"SELECT map -> 'key1' FROM \"TestEntity\" WHERE name = 'id'"))
|
||||||
|
.isEqualTo("value1");
|
||||||
|
assertThat(
|
||||||
|
getSingleResultFromNativeQuery(
|
||||||
|
"SELECT map -> 'key2' FROM \"TestEntity\" WHERE name = 'id'"))
|
||||||
|
.isEqualTo("value2");
|
||||||
|
|
||||||
|
executeNativeQuery("UPDATE \"TestEntity\" SET map = 'key3=>value3' WHERE name = 'id'");
|
||||||
|
|
||||||
|
assertThat(
|
||||||
|
getSingleResultFromNativeQuery(
|
||||||
|
"SELECT map -> 'key3' FROM \"TestEntity\" WHERE name = 'id'"))
|
||||||
|
.isEqualTo("value3");
|
||||||
|
|
||||||
|
executeNativeQuery("DELETE FROM \"TestEntity\" WHERE name = 'id'");
|
||||||
|
assertThrows(
|
||||||
|
NoResultException.class,
|
||||||
|
() ->
|
||||||
|
getSingleResultFromNativeQuery(
|
||||||
|
"SELECT map -> 'key3' FROM \"TestEntity\" WHERE name = 'id'"));
|
||||||
|
}
|
||||||
|
|
||||||
|
private static Object getSingleResultFromNativeQuery(String sql) {
|
||||||
|
return jpaTm()
|
||||||
|
.transact(() -> jpaTm().getEntityManager().createNativeQuery(sql).getSingleResult());
|
||||||
|
}
|
||||||
|
|
||||||
|
private static Object executeNativeQuery(String sql) {
|
||||||
|
return jpaTm()
|
||||||
|
.transact(() -> jpaTm().getEntityManager().createNativeQuery(sql).executeUpdate());
|
||||||
|
}
|
||||||
|
|
||||||
|
@Entity(name = "TestEntity") // Override entity name to avoid the nested class reference.
|
||||||
|
private static class TestEntity extends ImmutableObject {
|
||||||
|
|
||||||
|
@Id String name = "id";
|
||||||
|
|
||||||
|
@Type(type = "google.registry.persistence.MapUserType")
|
||||||
|
Map<String, String> map;
|
||||||
|
|
||||||
|
private TestEntity() {}
|
||||||
|
|
||||||
|
private TestEntity(Map<String, String> map) {
|
||||||
|
this.map = map;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
|
@ -16,6 +16,7 @@ package google.registry.schema.integration;
|
||||||
|
|
||||||
import google.registry.model.registry.RegistryLockDaoTest;
|
import google.registry.model.registry.RegistryLockDaoTest;
|
||||||
import google.registry.model.transaction.JpaTestRules.JpaIntegrationTestRule;
|
import google.registry.model.transaction.JpaTestRules.JpaIntegrationTestRule;
|
||||||
|
import google.registry.persistence.MapUserTypeTest;
|
||||||
import google.registry.schema.cursor.CursorDaoTest;
|
import google.registry.schema.cursor.CursorDaoTest;
|
||||||
import google.registry.schema.tld.PremiumListDaoTest;
|
import google.registry.schema.tld.PremiumListDaoTest;
|
||||||
import google.registry.schema.tld.PremiumListUtilsTest;
|
import google.registry.schema.tld.PremiumListUtilsTest;
|
||||||
|
@ -45,6 +46,7 @@ import org.junit.runners.Suite.SuiteClasses;
|
||||||
CreateReservedListCommandTest.class,
|
CreateReservedListCommandTest.class,
|
||||||
CursorDaoTest.class,
|
CursorDaoTest.class,
|
||||||
CreatePremiumListActionTest.class,
|
CreatePremiumListActionTest.class,
|
||||||
|
MapUserTypeTest.class,
|
||||||
PremiumListDaoTest.class,
|
PremiumListDaoTest.class,
|
||||||
PremiumListUtilsTest.class,
|
PremiumListUtilsTest.class,
|
||||||
RegistryLockDaoTest.class,
|
RegistryLockDaoTest.class,
|
||||||
|
|
|
@ -0,0 +1,15 @@
|
||||||
|
-- Copyright 2020 The Nomulus Authors. All Rights Reserved.
|
||||||
|
--
|
||||||
|
-- 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.
|
||||||
|
|
||||||
|
CREATE EXTENSION IF NOT EXISTS hstore WITH SCHEMA public;
|
|
@ -16,6 +16,20 @@ SET xmloption = content;
|
||||||
SET client_min_messages = warning;
|
SET client_min_messages = warning;
|
||||||
SET row_security = off;
|
SET row_security = off;
|
||||||
|
|
||||||
|
--
|
||||||
|
-- Name: hstore; Type: EXTENSION; Schema: -; Owner: -
|
||||||
|
--
|
||||||
|
|
||||||
|
CREATE EXTENSION IF NOT EXISTS hstore WITH SCHEMA public;
|
||||||
|
|
||||||
|
|
||||||
|
--
|
||||||
|
-- Name: EXTENSION hstore; Type: COMMENT; Schema: -; Owner: -
|
||||||
|
--
|
||||||
|
|
||||||
|
COMMENT ON EXTENSION hstore IS 'data type for storing sets of (key, value) pairs';
|
||||||
|
|
||||||
|
|
||||||
SET default_tablespace = '';
|
SET default_tablespace = '';
|
||||||
|
|
||||||
SET default_with_oids = false;
|
SET default_with_oids = false;
|
||||||
|
|
Loading…
Add table
Add a link
Reference in a new issue