From 659fa868d8cd2bbef65f57e3dc74275d36031c28 Mon Sep 17 00:00:00 2001 From: felmer <felmer> Date: Wed, 16 May 2012 11:10:05 +0000 Subject: [PATCH] SP-39 reading and writing timestamp SVN: 25276 --- .../server/task/MaterialReportingTask.java | 91 +++++++++++++++++-- .../task/MaterialReportingTaskTest.java | 56 +++++++++++- 2 files changed, 134 insertions(+), 13 deletions(-) diff --git a/openbis/source/java/ch/systemsx/cisd/openbis/generic/server/task/MaterialReportingTask.java b/openbis/source/java/ch/systemsx/cisd/openbis/generic/server/task/MaterialReportingTask.java index 315e0c0d048..5222d54581f 100644 --- a/openbis/source/java/ch/systemsx/cisd/openbis/generic/server/task/MaterialReportingTask.java +++ b/openbis/source/java/ch/systemsx/cisd/openbis/generic/server/task/MaterialReportingTask.java @@ -37,7 +37,9 @@ import java.util.Properties; import java.util.Set; import java.util.TreeMap; +import org.apache.commons.lang.time.DateFormatUtils; import org.apache.log4j.Logger; +import org.springframework.dao.IncorrectResultSizeDataAccessException; import org.springframework.jdbc.core.BatchPreparedStatementSetter; import org.springframework.jdbc.core.ColumnMapRowMapper; import org.springframework.jdbc.core.JdbcTemplate; @@ -52,7 +54,9 @@ import ch.systemsx.cisd.common.filesystem.FileUtilities; import ch.systemsx.cisd.common.logging.LogCategory; import ch.systemsx.cisd.common.logging.LogFactory; import ch.systemsx.cisd.common.maintenance.IMaintenanceTask; +import ch.systemsx.cisd.common.utilities.ITimeProvider; import ch.systemsx.cisd.common.utilities.PropertyUtils; +import ch.systemsx.cisd.common.utilities.SystemTimeProvider; import ch.systemsx.cisd.dbmigration.SimpleDatabaseConfigurationContext; import ch.systemsx.cisd.openbis.generic.server.CommonServiceProvider; import ch.systemsx.cisd.openbis.generic.server.ICommonServerForInternalUse; @@ -70,6 +74,7 @@ import ch.systemsx.cisd.openbis.generic.shared.basic.dto.PropertyType; import ch.systemsx.cisd.openbis.generic.shared.basic.dto.SearchCriteriaConnection; import ch.systemsx.cisd.openbis.generic.shared.dto.SessionContextDTO; import ch.systemsx.cisd.openbis.generic.shared.util.DataTypeUtils; +import ch.systemsx.cisd.openbis.generic.shared.util.SimplePropertyValidator.SupportedDatePattern; /** * Task which feeds a reporting database with recently added/changed Materials. @@ -78,6 +83,15 @@ import ch.systemsx.cisd.openbis.generic.shared.util.DataTypeUtils; */ public class MaterialReportingTask implements IMaintenanceTask { + @Private + static final String READ_TIMESTAMP_SQL_KEY = "read-timestamp-sql"; + + @Private + static final String UPDATE_TIMESTAMP_SQL_KEY = "update-timestamp-sql"; + + @Private + static final String INSERT_TIMESTAMP_SQL_KEY = "insert-timestamp-sql"; + @Private static final String MAPPING_FILE_KEY = "mapping-file"; @@ -347,29 +361,38 @@ public class MaterialReportingTask implements IMaintenanceTask private final ICommonServerForInternalUse server; + private final ITimeProvider timeProvider; + private Map<String, MappingInfo> mapping; private SimpleDatabaseConfigurationContext dbConfigurationContext; private JdbcTemplate jdbcTemplate; + private String readTimestampSql; + + private String insertTimestampSql; + + private String updateTimestampSql; + public MaterialReportingTask() { - this(CommonServiceProvider.getCommonServer()); + this(CommonServiceProvider.getCommonServer(), SystemTimeProvider.SYSTEM_TIME_PROVIDER); } - public MaterialReportingTask(ICommonServerForInternalUse server) + public MaterialReportingTask(ICommonServerForInternalUse server, ITimeProvider timeProvider) { this.server = server; + this.timeProvider = timeProvider; } public void setUp(String pluginName, Properties properties) { dbConfigurationContext = new SimpleDatabaseConfigurationContext(properties); - // String readTimestampSql = PropertyUtils.getMandatoryProperty(properties, - // "read-timestamp-sql"); - // String writeTimestampSql = PropertyUtils.getMandatoryProperty(properties, - // "write-timestamp-sql"); + readTimestampSql = PropertyUtils.getMandatoryProperty(properties, READ_TIMESTAMP_SQL_KEY); + updateTimestampSql = + PropertyUtils.getMandatoryProperty(properties, UPDATE_TIMESTAMP_SQL_KEY); + insertTimestampSql = properties.getProperty(INSERT_TIMESTAMP_SQL_KEY, updateTimestampSql); String mappingFileName = PropertyUtils.getMandatoryProperty(properties, MAPPING_FILE_KEY); mapping = readMappingFile(mappingFileName); Map<String, Map<String, PropertyType>> materialTypes = getMaterialTypes(); @@ -393,6 +416,7 @@ public class MaterialReportingTask implements IMaintenanceTask mappingInfo.injectDataTypeCodes(columns, propertyTypes); } jdbcTemplate = new JdbcTemplate(dbConfigurationContext.getDataSource()); + checkTimestampReadingWriting(); } public void execute() @@ -415,6 +439,7 @@ public class MaterialReportingTask implements IMaintenanceTask addOrUpdate(mappingInfo, materials); } } + writeTimestamp(new Date(timeProvider.getTimeInMilliseconds())); operationLog.info("Reporting finished."); } @@ -551,7 +576,7 @@ public class MaterialReportingTask implements IMaintenanceTask new DetailedSearchCriterion( DetailedSearchField .createAttributeField(MaterialAttributeSearchFieldKind.MODIFICATION_DATE), - CompareType.MORE_THAN_OR_EQUAL, readTimestamp(), "0"); + CompareType.MORE_THAN_OR_EQUAL, readTimestamp()); criteria.setCriteria(Arrays.asList(criterion)); criteria.setConnection(SearchCriteriaConnection.MATCH_ALL); List<Material> materials = server.searchForMaterials(sessionToken, criteria); @@ -570,9 +595,59 @@ public class MaterialReportingTask implements IMaintenanceTask return result; } + private void checkTimestampReadingWriting() + { + Date timestamp; + try + { + timestamp = tryToReadTimestamp(); + } catch (Exception ex) + { + throw new ConfigurationFailureException( + "Couldn't get timestamp from report database. Property '" + + READ_TIMESTAMP_SQL_KEY + "' could be invalid.", ex); + } + try + { + writeTimestamp(timestamp == null ? new Date(0) : timestamp); + } catch (Exception ex) + { + throw new ConfigurationFailureException( + "Couldn't save timestamp to report database. Property '" + + INSERT_TIMESTAMP_SQL_KEY + "' or '" + UPDATE_TIMESTAMP_SQL_KEY + + "' could be invalid.", ex); + } + } + private String readTimestamp() { - return "2012-02-20 10:33:44.6667"; + Date timestamp = tryToReadTimestamp(); + return timestamp == null ? "1970-01-01" : DateFormatUtils.format(timestamp, + SupportedDatePattern.CANONICAL_DATE_PATTERN.getPattern()); + } + + private Date tryToReadTimestamp() + { + try + { + return (Date) jdbcTemplate.queryForObject(readTimestampSql, Date.class); + } catch (IncorrectResultSizeDataAccessException ex) + { + int actualSize = ex.getActualSize(); + if (actualSize == 0) + { + return null; + } + throw ex; + } + } + + private void writeTimestamp(Date newTimestamp) + { + String sql = tryToReadTimestamp() == null ? insertTimestampSql : updateTimestampSql; + jdbcTemplate.update(sql, new Object[] + { newTimestamp }, new int[] + { Types.TIMESTAMP }); } @Private diff --git a/openbis/sourceTest/java/ch/systemsx/cisd/openbis/generic/server/task/MaterialReportingTaskTest.java b/openbis/sourceTest/java/ch/systemsx/cisd/openbis/generic/server/task/MaterialReportingTaskTest.java index e3cfb0631d7..625b97657d3 100644 --- a/openbis/sourceTest/java/ch/systemsx/cisd/openbis/generic/server/task/MaterialReportingTaskTest.java +++ b/openbis/sourceTest/java/ch/systemsx/cisd/openbis/generic/server/task/MaterialReportingTaskTest.java @@ -43,11 +43,15 @@ import ch.systemsx.cisd.common.exceptions.ConfigurationFailureException; import ch.systemsx.cisd.common.exceptions.EnvironmentFailureException; import ch.systemsx.cisd.common.filesystem.FileUtilities; import ch.systemsx.cisd.common.test.RecordingMatcher; +import ch.systemsx.cisd.common.utilities.ITimeProvider; import ch.systemsx.cisd.dbmigration.DatabaseConfigurationContext; import ch.systemsx.cisd.openbis.generic.server.ICommonServerForInternalUse; import ch.systemsx.cisd.openbis.generic.server.task.MaterialReportingTask.MappingInfo; +import ch.systemsx.cisd.openbis.generic.shared.basic.dto.CompareType; import ch.systemsx.cisd.openbis.generic.shared.basic.dto.DataTypeCode; import ch.systemsx.cisd.openbis.generic.shared.basic.dto.DetailedSearchCriteria; +import ch.systemsx.cisd.openbis.generic.shared.basic.dto.DetailedSearchCriterion; +import ch.systemsx.cisd.openbis.generic.shared.basic.dto.DetailedSearchFieldKind; import ch.systemsx.cisd.openbis.generic.shared.basic.dto.Material; import ch.systemsx.cisd.openbis.generic.shared.basic.dto.MaterialType; import ch.systemsx.cisd.openbis.generic.shared.basic.dto.builders.MaterialBuilder; @@ -78,12 +82,15 @@ public class MaterialReportingTaskTest extends AbstractFileSystemTestCase private Properties properties; + private ITimeProvider timeProvider; + @BeforeMethod public void setUpMocks() throws Exception { context = new Mockery(); server = context.mock(ICommonServerForInternalUse.class); - materialReportingTask = new MaterialReportingTask(server); + timeProvider = context.mock(ITimeProvider.class); + materialReportingTask = new MaterialReportingTask(server, timeProvider); dbConfigContext = new DatabaseConfigurationContext(); dbConfigContext.setDatabaseEngineCode("postgresql"); @@ -94,12 +101,19 @@ public class MaterialReportingTaskTest extends AbstractFileSystemTestCase dropTestDatabase(); createTestDatabase(); createTables( + "create table timestamp (timestamp timestamp)", "create table report1 (id bigint, code varchar(20), description varchar(200))", "create table report2 (code varchar(20), rank integer, greetings varchar(200), " + "size double precision, organism varchar(100), material varchar(30), timestamp timestamp)"); dbConfigContext.closeConnections(); mappingFile = new File(workingDirectory, "mapping-file.txt"); properties = new Properties(); + properties.setProperty(MaterialReportingTask.READ_TIMESTAMP_SQL_KEY, + "select timestamp from timestamp"); + properties.setProperty(MaterialReportingTask.INSERT_TIMESTAMP_SQL_KEY, + "insert into timestamp values(?)"); + properties.setProperty(MaterialReportingTask.UPDATE_TIMESTAMP_SQL_KEY, + "update timestamp set timestamp = ?"); properties.setProperty("database-driver", "org.postgresql.Driver"); properties.setProperty("database-url", "jdbc:postgresql://localhost/" + databaseName); properties.setProperty("database-username", "postgres"); @@ -509,6 +523,9 @@ public class MaterialReportingTaskTest extends AbstractFileSystemTestCase one(server).searchForMaterials(with(SESSION_TOKEN), with(criteriaRecorder)); will(returnValue(Arrays.asList(m1, m2, m3))); + + one(timeProvider).getTimeInMilliseconds(); + will(returnValue(24L * 3600L * 1000L * 60L)); } }); @@ -521,6 +538,15 @@ public class MaterialReportingTaskTest extends AbstractFileSystemTestCase + "size=1.00000005E7, timestamp=1970-02-03 01:00:00.0}, " + "{code=M3, greetings=hello, material=null, organism=null, rank=null, " + "size=null, timestamp=null}]", result.toString()); + assertEquals("[{timestamp=1970-03-02 01:00:00.0}]", loadTable("timestamp", false) + .toString()); + DetailedSearchCriterion detailedSearchCriterion = + criteriaRecorder.recordedObject().getCriteria().get(0); + assertEquals(CompareType.MORE_THAN_OR_EQUAL, detailedSearchCriterion.getType()); + assertEquals("MODIFICATION_DATE", detailedSearchCriterion.getField().getAttributeCode()); + assertEquals(DetailedSearchFieldKind.ATTRIBUTE, detailedSearchCriterion.getField() + .getKind()); + assertEquals("1970-01-01 01:00:00 +0100", detailedSearchCriterion.getValue()); context.assertIsSatisfied(); } @@ -549,7 +575,8 @@ public class MaterialReportingTaskTest extends AbstractFileSystemTestCase new MaterialBuilder().code("M4").type("T2").property("P2", "hi").getMaterial(); final RecordingMatcher<DetailedSearchCriteria> criteriaRecorder = new RecordingMatcher<DetailedSearchCriteria>(); - final Sequence sequence = context.sequence("materials"); + final Sequence searchMaterialSequence = context.sequence("materials"); + final Sequence timeSequence = context.sequence("time"); context.checking(new Expectations() { { @@ -560,11 +587,19 @@ public class MaterialReportingTaskTest extends AbstractFileSystemTestCase one(server).searchForMaterials(with(SESSION_TOKEN), with(criteriaRecorder)); will(returnValue(Arrays.asList(m1, m2, m3))); - inSequence(sequence); + inSequence(searchMaterialSequence); one(server).searchForMaterials(with(SESSION_TOKEN), with(criteriaRecorder)); will(returnValue(Arrays.asList(m1, m2v2, m4))); - inSequence(sequence); + inSequence(searchMaterialSequence); + + one(timeProvider).getTimeInMilliseconds(); + will(returnValue(24L * 3600L * 1000L * 60L)); + inSequence(timeSequence); + + one(timeProvider).getTimeInMilliseconds(); + will(returnValue(24L * 3600L * 1000L * 62L)); + inSequence(timeSequence); } }); @@ -580,6 +615,12 @@ public class MaterialReportingTaskTest extends AbstractFileSystemTestCase + "rank=null, size=null, timestamp=null}, " + "{code=M4, greetings=hi, material=null, organism=null, " + "rank=null, size=null, timestamp=null}]", result.toString()); + assertEquals("[{timestamp=1970-03-04 01:00:00.0}]", loadTable("timestamp", false) + .toString()); + assertEquals("1970-01-01 01:00:00 +0100", criteriaRecorder.getRecordedObjects().get(0) + .getCriteria().get(0).getValue()); + assertEquals("1970-03-02 01:00:00 +0100", criteriaRecorder.getRecordedObjects().get(1) + .getCriteria().get(0).getValue()); context.assertIsSatisfied(); } @@ -614,9 +655,14 @@ public class MaterialReportingTaskTest extends AbstractFileSystemTestCase } private List<?> loadTable(String tableName) + { + return loadTable(tableName, true); + } + + private List<?> loadTable(String tableName, boolean orderByCode) { return new JdbcTemplate(dbConfigContext.getDataSource()).query("select * from " + tableName - + " order by code", new ColumnMapRowMapper() + + (orderByCode ? " order by code" : ""), new ColumnMapRowMapper() { @SuppressWarnings("rawtypes") @Override -- GitLab