diff --git a/conf/smart-default.xml b/conf/smart-default.xml index 27c480859d6..dbfdfdf57f0 100644 --- a/conf/smart-default.xml +++ b/conf/smart-default.xml @@ -367,15 +367,6 @@ - - smart.metastore.mysql.legacy.enabled - false - - Whether to enable support for versions of MySQL lesser than 5.7. - Warning: it requires admin privileges for ssm user to be able to configure the db. - - - smart.ignore.path.templates @@ -393,45 +384,7 @@ - smart.access.count.day.tables.num - 30 - - The max number of access count per day tables in the Metastore. - - - - - smart.access.count.hour.tables.num - 48 - - The max number of access count per hour tables in the Metastore. - It should be at least 24 to cover the full day to be successfully - aggregated into a per day access count table. - - - - - smart.access.count.minute.tables.num - 120 - - The max number of access count per minute tables in the Metastore. - It should be at least 60 to cover the full hour to be successfully - aggregated into a per hour access count table. - - - - - smart.access.count.second.tables.num - 30 - - The max number of access count per second tables in the Metastore. - It should be at least 60000/'smart.access.count.aggregation.interval.ms' - to cover the full minute to be successfully aggregated into a per minute access count table. - - - - - smart.access.event.fetch.interval.ms + smart.file.access.event.fetch.interval.ms 1000 The interval in milliseconds between access event fetches. @@ -479,7 +432,7 @@ - smart.access.count.aggregation.interval.ms + smart.file.access.count.aggregation.interval.ms 5000 The interval in milliseconds that is covered by single second-granularity access count table. @@ -578,17 +531,22 @@ - smart.access.count.aggregator.failover - SUBMIT_NEW_FAILED_EVENTS_LATER + smart.file.access.count.aggregator.failover.retry.count + 60 + + Maximum number of attempts to save file access events + + + + + smart.file.access.count.aggregator.failover + SUBMIT_FAILED_EVENTS_WITH_RETRY Failover strategy for file access events aggregator. Possible values: - DROP_EVENTS - drop file access events that caused exception. FAIL - throw exception, no failover. - SUBMIT_FAILED_EVENTS_LATER - save all file access events that caused exception - for later submission. Should be used carefully, because in case of repeated - exceptions can potentially cause OOM error. - SUBMIT_NEW_FAILED_EVENTS_LATER - save new file access events that caused exception - for later submission and drop old failing events. + SUBMIT_FAILED_EVENTS_WITH_RETRY - save all file access events that caused exception + for later submission with max attempts less or equals than smart.access.count.aggregator.failover.retry.count + diff --git a/docs/admin-user-guide.md b/docs/admin-user-guide.md index 538511651b1..1d0785fdfb0 100755 --- a/docs/admin-user-guide.md +++ b/docs/admin-user-guide.md @@ -78,23 +78,21 @@ Table-4 Condition Ingredient Table-5 Object Properties -| Object | Property | Description | -|----------|----------------------------------|---------------------------------------------------------------------------------------------| -| | age | The time span from last modification moment to now | -| | atime | The last access time | -| | blocksize | The block size of the file | -| | inCache | The file is in cache storage | -| | isDir | The file is a directory | -| | length | Length of the file. Currently, only pure digital is supported, which indicates bytes number.| -| file | path | The file path in HDFS | -| | mtime | The last modification time of the file | -| | unsynced | The file is not synced | -| | storagePolicy | Storage policy of file | -| | accessCount(Time Interval) | The access counts during the last time interval | -| | accessCountTop(interval,N ) | The topmost N for access counts during the last time interval | -| | accessCountBottom(interval,N) | The bottommost N for access counts during the last time interval | -| | accessCountTopOnStoragePolicy(interval,N,$StoragePolicy") | The topmost N for access counts with regard to a storage policy.The supported HDFS storage policies are COLD,WARM,HOT,ONE_SSD,ALL_SSD,LAZY_PERSIST | -| | accessCountBottomOnStoragePolicy(interval,N,$StoragePolicy") | The bottommost N for access counts with regard to a storage policy during the last time interval | +| Object | Property | Description | +|----------|--------------------------------------------------------------|----------------------------------------------------------------------------------------------------------------------------------------------------| +| | age | The time span from last modification moment to now | +| | atime | The last access time | +| | blocksize | The block size of the file | +| | inCache | The file is in cache storage | +| | isDir | The file is a directory | +| | length | Length of the file. Currently, only pure digital is supported, which indicates bytes number. | +| file | path | The file path in HDFS | +| | mtime | The last modification time of the file | +| | unsynced | The file is not synced | +| | storagePolicy | Storage policy of file | +| | accessCount(Time Interval) | The access counts during the last time interval | +| | accessCountTop(interval,N ) | The topmost N for access counts during the last time interval | +| | accessCountBottom(interval,N) | The bottommost N for access counts during the last time interval | Table-6 Commands diff --git a/smart-common/src/main/java/org/smartdata/conf/SmartConfKeys.java b/smart-common/src/main/java/org/smartdata/conf/SmartConfKeys.java index 366ce83b938..6363abc7b5d 100644 --- a/smart-common/src/main/java/org/smartdata/conf/SmartConfKeys.java +++ b/smart-common/src/main/java/org/smartdata/conf/SmartConfKeys.java @@ -79,41 +79,22 @@ public class SmartConfKeys { // Password which get from hadoop credentialProvider used for metastore connect public static final String SMART_METASTORE_PASSWORD = "smart.metastore.password"; - public static final String SMART_METASTORE_MIGRATION_CHANGELOG_PATH_KEY = - "smart.metastore.migration.liquibase.changelog.path"; - public static final String SMART_METASTORE_MIGRATION_CHANGELOG_PATH_DEFAULT = - "db/changelog/changelog-root.xml"; - public static final String SMART_METASTORE_LEGACY_MYSQL_SUPPORT_KEY = - "smart.metastore.mysql.legacy.enabled"; - public static final boolean SMART_METASTORE_LEGACY_MYSQL_SUPPORT_DEFAULT = false; - - public static final String SMART_ACCESS_COUNT_AGGREGATION_INTERVAL_MS = - "smart.access.count.aggregation.interval.ms"; - public static final int SMART_ACCESS_COUNT_AGGREGATION_INTERVAL_MS_DEFAULT = 5000; + public static final String SMART_METASTORE_MIGRATION_CHANGELOG_PATH_KEY = + "smart.metastore.migration.liquibase.changelog.path"; + public static final String SMART_METASTORE_MIGRATION_CHANGELOG_PATH_DEFAULT = + "db/changelog/changelog-root.xml"; public static final String SMART_ACCESS_COUNT_AGGREGATOR_FAILOVER_KEY = - "smart.access.count.aggregator.failover"; + "smart.file.access.count.aggregator.failover"; + public static final String SMART_ACCESS_COUNT_AGGREGATOR_FAILOVER_MAX_RETRIES_KEY = + "smart.file.access.count.aggregator.failover.retry.count"; + public static final int SMART_ACCESS_COUNT_AGGREGATOR_FAILOVER_MAX_RETRIES_DEFAULT = 60; - public static final String SMART_NUM_DAY_TABLES_TO_KEEP_KEY = - "smart.access.count.day.tables.num"; - public static final int SMART_NUM_DAY_TABLES_TO_KEEP_DEFAULT = 30; - - public static final String SMART_NUM_HOUR_TABLES_TO_KEEP_KEY = - "smart.access.count.hour.tables.num"; - public static final int SMART_NUM_HOUR_TABLES_TO_KEEP_DEFAULT = 48; - public static final int SMART_NUM_HOUR_TABLES_TO_KEEP_MIN = 24; - - public static final String SMART_NUM_MINUTE_TABLES_TO_KEEP_KEY = - "smart.access.count.minute.tables.num"; - public static final int SMART_NUM_MINUTE_TABLES_TO_KEEP_DEFAULT = 120; - public static final int SMART_NUM_MINUTE_TABLES_TO_KEEP_MIN = 60; - - public static final String SMART_NUM_SECOND_TABLES_TO_KEEP_KEY = - "smart.access.count.second.tables.num"; - public static final int SMART_NUM_SECOND_TABLES_TO_KEEP_DEFAULT = 30; + public static final String SMART_FILE_ACCESS_PARTITIONS_RETENTION_POLICY_KEY = + "smart.file.access.partition.retention.policy"; public static final String SMART_ACCESS_EVENT_FETCH_INTERVAL_MS_KEY = - "smart.access.event.fetch.interval.ms"; + "smart.file.access.event.fetch.interval.ms"; public static final long SMART_ACCESS_EVENT_FETCH_INTERVAL_MS_DEFAULT = 1000L; public static final String SMART_CACHED_FILE_FETCH_INTERVAL_MS_KEY = @@ -124,7 +105,12 @@ public class SmartConfKeys { "smart.namespace.fetch.interval.ms"; public static final long SMART_NAMESPACE_FETCH_INTERVAL_MS_DEFAULT = 1L; - // StatesManager + // File access partitions + public static final String SMART_FILE_ACCESS_PARTITIONS_RETENTION_COUNT_KEY = + "smart.file.access.partition.retention.count"; + public static final int SMART_FILE_ACCESS_PARTITIONS_RETENTION_COUNT_DEFAULT = 24; + + // StatesManager // RuleManager public static final String SMART_RULE_EXECUTORS_KEY = "smart.rule.executors"; diff --git a/smart-common/src/main/java/org/smartdata/model/request/FileAccessInfoSearchRequest.java b/smart-common/src/main/java/org/smartdata/model/request/FileAccessInfoSearchRequest.java index 74fb02cc231..171cfed1676 100644 --- a/smart-common/src/main/java/org/smartdata/model/request/FileAccessInfoSearchRequest.java +++ b/smart-common/src/main/java/org/smartdata/model/request/FileAccessInfoSearchRequest.java @@ -24,7 +24,6 @@ import org.smartdata.model.TimeInterval; import java.util.List; -import java.util.Set; @Data @Builder @@ -34,7 +33,6 @@ public class FileAccessInfoSearchRequest { private final List ids; private final String pathLike; private final TimeInterval lastAccessedTime; - private final Set accessCountTables; public static FileAccessInfoSearchRequest noFilters() { return builder().build(); diff --git a/smart-engine/pom.xml b/smart-engine/pom.xml index e6d298916e7..f1a1b9f8eb3 100644 --- a/smart-engine/pom.xml +++ b/smart-engine/pom.xml @@ -182,6 +182,16 @@ log4j-slf4j-impl test + + org.testcontainers + jdbc + test + + + org.testcontainers + postgresql + test + diff --git a/smart-engine/src/main/java/org/smartdata/server/engine/StatesManager.java b/smart-engine/src/main/java/org/smartdata/server/engine/StatesManager.java index dbb0e9e765b..2b8a8fd6dc9 100644 --- a/smart-engine/src/main/java/org/smartdata/server/engine/StatesManager.java +++ b/smart-engine/src/main/java/org/smartdata/server/engine/StatesManager.java @@ -25,9 +25,13 @@ import org.smartdata.conf.ReconfigurableRegistry; import org.smartdata.conf.ReconfigureException; import org.smartdata.conf.SmartConfKeys; -import org.smartdata.metastore.MetaStoreException; -import org.smartdata.metastore.dao.accesscount.AccessCountTableManager; -import org.smartdata.metastore.model.AccessCountTable; +import org.smartdata.metastore.accesscount.DbAccessEventAggregator; +import org.smartdata.metastore.accesscount.FileAccessManager; +import org.smartdata.metastore.accesscount.failover.AccessCountFailoverFactory; +import org.smartdata.metastore.partition.FileAccessPartitionManagerImpl; +import org.smartdata.metastore.partition.FileAccessPartitionService; +import org.smartdata.metastore.partition.cleanup.FileAccessPartitionRetentionPolicyExecutorFactory; +import org.smartdata.metastore.transaction.TransactionRunner; import org.smartdata.metrics.FileAccessEvent; import org.smartdata.metrics.FileAccessEventSource; import org.smartdata.metrics.impl.MetricsFactory; @@ -40,6 +44,8 @@ import java.util.concurrent.Executors; import java.util.concurrent.ScheduledExecutorService; +import static org.springframework.transaction.annotation.Isolation.SERIALIZABLE; + /** * Polls metrics and events from NameNode. */ @@ -47,12 +53,14 @@ public class StatesManager extends AbstractService implements Reconfigurable { private final ServerContext serverContext; private ScheduledExecutorService executorService; - private AccessCountTableManager accessCountTableManager; + @Getter + private FileAccessManager fileAccessManager; private AccessEventFetcher accessEventFetcher; private FileAccessEventSource fileAccessEventSource; @Getter private CachedFilesManager cachedFilesManager; private AbstractService statesUpdaterService; + private FileAccessPartitionService fileAccessPartitionService; private PathChecker pathChecker; private volatile boolean working = false; @@ -69,18 +77,39 @@ public StatesManager(ServerContext context) { @Override public void init() throws IOException { LOG.info("Initializing ..."); - this.executorService = Executors.newScheduledThreadPool(4); - this.accessCountTableManager = new AccessCountTableManager( - serverContext.getMetaStore(), executorService, serverContext.getConf()); + this.executorService = Executors.newScheduledThreadPool(5); + TransactionRunner transactionRunner = + new TransactionRunner(serverContext.getMetaStore().transactionManager()); + transactionRunner.setIsolationLevel(SERIALIZABLE); + this.fileAccessManager = new FileAccessManager( + transactionRunner, + serverContext.getMetaStore().accessCountEventDao(), + serverContext.getMetaStore().cacheFileDao()); this.fileAccessEventSource = MetricsFactory.createAccessEventSource(serverContext.getConf()); + AccessCountFailoverFactory accessCountFailoverFactory = + new AccessCountFailoverFactory(serverContext.getConf()); + DbAccessEventAggregator accessEventAggregator = new DbAccessEventAggregator( + serverContext.getMetaStore().fileInfoDao(), + fileAccessManager, + accessCountFailoverFactory.create()); this.accessEventFetcher = new AccessEventFetcher( serverContext.getConf(), - accessCountTableManager.getAccessEventAggregator(), + accessEventAggregator, executorService, fileAccessEventSource.getCollector()); this.pathChecker = new PathChecker(serverContext.getConf()); this.cachedFilesManager = new CachedFilesManager(serverContext.getMetaStore().cacheFileDao()); + FileAccessPartitionRetentionPolicyExecutorFactory + fileAccessPartitionRetentionPolicyExecutorFactory = + new FileAccessPartitionRetentionPolicyExecutorFactory( + serverContext.getMetaStore()); + this.fileAccessPartitionService = new FileAccessPartitionService( + executorService, + new FileAccessPartitionManagerImpl(serverContext.getMetaStore()), + fileAccessPartitionRetentionPolicyExecutorFactory.createPolicyExecutor( + serverContext.getConf()) + ); initStatesUpdaterService(); if (statesUpdaterService == null) { @@ -105,6 +134,7 @@ public boolean inSafeMode() { @Override public void start() throws IOException { LOG.info("Starting ..."); + fileAccessPartitionService.start(); accessEventFetcher.start(); if (statesUpdaterService != null) { statesUpdaterService.start(); @@ -118,6 +148,9 @@ public void stop() throws IOException { working = false; LOG.info("Stopping ..."); + if (fileAccessPartitionService != null) { + fileAccessPartitionService.stop(); + } if (accessEventFetcher != null) { accessEventFetcher.stop(); } @@ -134,10 +167,6 @@ public void stop() throws IOException { LOG.info("Stopped."); } - public List getTablesForLast(long timeInMills) throws MetaStoreException { - return accessCountTableManager.getTablesForLast(timeInMills); - } - public void reportFileAccessEvent(FileAccessEvent event) { String path = event.getPath(); path = path + (path.endsWith("/") ? "" : "/"); @@ -203,8 +232,4 @@ private synchronized void initStatesUpdaterService() { LOG.info("", t); } } - - public AccessCountTableManager getAccessCountTableManager() { - return accessCountTableManager; - } } diff --git a/smart-engine/src/main/java/org/smartdata/server/engine/data/AccessEventFetcher.java b/smart-engine/src/main/java/org/smartdata/server/engine/data/AccessEventFetcher.java index 28305fe6763..e50a54fa293 100644 --- a/smart-engine/src/main/java/org/smartdata/server/engine/data/AccessEventFetcher.java +++ b/smart-engine/src/main/java/org/smartdata/server/engine/data/AccessEventFetcher.java @@ -20,7 +20,7 @@ import org.apache.hadoop.conf.Configuration; import org.slf4j.Logger; import org.slf4j.LoggerFactory; -import org.smartdata.metastore.dao.accesscount.AccessEventAggregator; +import org.smartdata.metastore.accesscount.AccessEventAggregator; import org.smartdata.metrics.FileAccessEvent; import org.smartdata.metrics.FileAccessEventCollector; @@ -55,10 +55,8 @@ public AccessEventFetcher( } public void start() { - long current = System.currentTimeMillis(); - long toWait = fetchInterval - (current % fetchInterval); this.scheduledFuture = scheduledExecutorService.scheduleAtFixedRate( - fetchTask, toWait, fetchInterval, TimeUnit.MILLISECONDS); + fetchTask, 0, fetchInterval, TimeUnit.MILLISECONDS); } public void stop() { @@ -71,8 +69,8 @@ private static class FetchTask implements Runnable { private final AccessEventAggregator accessEventAggregator; private final FileAccessEventCollector collector; - public FetchTask( - AccessEventAggregator accessEventAggregator, FileAccessEventCollector collector) { + public FetchTask(AccessEventAggregator accessEventAggregator, + FileAccessEventCollector collector) { this.accessEventAggregator = accessEventAggregator; this.collector = collector; } diff --git a/smart-engine/src/main/java/org/smartdata/server/engine/rule/RuleExecutor.java b/smart-engine/src/main/java/org/smartdata/server/engine/rule/RuleExecutor.java index ba2dc5251ba..1f98fde0db1 100644 --- a/smart-engine/src/main/java/org/smartdata/server/engine/rule/RuleExecutor.java +++ b/smart-engine/src/main/java/org/smartdata/server/engine/rule/RuleExecutor.java @@ -24,7 +24,6 @@ import org.smartdata.exception.QueueFullException; import org.smartdata.metastore.MetaStore; import org.smartdata.metastore.MetaStoreException; -import org.smartdata.metastore.model.AccessCountTable; import org.smartdata.model.CmdletDescriptor; import org.smartdata.model.RuleInfo; import org.smartdata.model.RuleState; @@ -43,7 +42,9 @@ import java.util.regex.Matcher; import java.util.regex.Pattern; -/** Execute rule queries and return result. */ +/** + * Execute rule queries and return result. + */ public class RuleExecutor implements Runnable { private static final Logger LOG = LoggerFactory.getLogger(RuleExecutor.class.getName()); @@ -187,138 +188,38 @@ private void genVirtualAccessCountTableValue(List parameters, boolean to executionCtx.setProperty(var, count == null ? 0L : count); } - public String genVirtualAccessCountTableTopValueOnStoragePolicy(List parameters) { - genVirtualAccessCountTableValueOnStoragePolicy(parameters, true); - return null; - } - - public String genVirtualAccessCountTableBottomValueOnStoragePolicy(List parameters) { - genVirtualAccessCountTableValueOnStoragePolicy(parameters, false); - return null; - } - - private void genVirtualAccessCountTableValueOnStoragePolicy(List parameters, - boolean top) { - List paraList = (List) parameters.get(0); - String table = (String) parameters.get(1); - String var = (String) parameters.get(2); - Long num = (Long) paraList.get(1); - String storage = ((String) paraList.get(2)).toUpperCase(); - String sqlsub; - if (storage.equals("CACHE")) { - sqlsub = String.format("SELECT %s.fid, %s.count FROM %s LEFT JOIN cached_file ON " - + "(%s.fid = cached_file.fid)", table, table, table, table); - } else { - Integer id = null; - try { - id = metastore.getStoragePolicyID(storage); - } catch (Exception e) { - // Ignore - } - if (id == null) { - id = -1; // safe return - } - sqlsub = String.format("SELECT %s.fid, %s.count FROM %s LEFT JOIN file ON " - + "(%s.fid = file.fid) WHERE file.sid = %d", - table, table, table, table, id); - } - - String sql0 = String.format( - "SELECT %s(count) FROM ( SELECT * FROM (%s) AS %s ORDER BY count %sLIMIT %d ) AS %s;", - top ? "min" : "max", - sqlsub, - table + "_AL1_TMP", - top ? "DESC " : "", - num, - table + "_AL2_TMP"); - Long count = null; - try { - count = metastore.queryForLong(sql0); - } catch (MetaStoreException e) { - LOG.error(String.format("Get %s access count on storage [%s] from table '%s' error [%s].", - top ? "top" : "bottom", storage, table, sql0), e); - } - executionCtx.setProperty(var, count == null ? 0L : count); - } - public String genVirtualAccessCountTable(List parameters) { List paraList = (List) parameters.get(0); String newTable = (String) parameters.get(1); - Long interval = (Long) paraList.get(0); + long interval = paraList.isEmpty() ? 0L : (long) paraList.get(0); String countFilter = ""; - List tableNames = getAccessCountTablesDuringLast(interval); - return generateSQL(tableNames, newTable, countFilter, metastore); + long currentTimeMillis = System.currentTimeMillis(); + return generateSQL(newTable, countFilter, metastore, currentTimeMillis - interval, + currentTimeMillis); } @VisibleForTesting static String generateSQL( - List tableNames, String newTable, String countFilter, MetaStore adapter) { + String newTable, + String countFilter, + MetaStore adapter, + long startTime, + long endTime) { String sqlFinal, sqlCreate; - if (tableNames.size() <= 1) { - String tableName = tableNames.isEmpty() ? "blank_access_count_info" : tableNames.get(0); - sqlCreate = "CREATE TABLE " + newTable + "(fid INTEGER NOT NULL, count INTEGER NOT NULL);"; - try { - adapter.execute(sqlCreate); - } catch (MetaStoreException e) { - LOG.error("Cannot create table " + newTable, e); - } - sqlFinal = "INSERT INTO " + newTable + " SELECT * FROM " + tableName + ";"; - } else { - String sqlPrefix = "SELECT fid, SUM(count) AS count FROM (\n"; - String sqlUnion = "SELECT fid, count FROM " + tableNames.get(0) + " \n"; - for (int i = 1; i < tableNames.size(); i++) { - sqlUnion += "UNION ALL\n" + "SELECT fid, count FROM " + tableNames.get(i) + " \n"; - } - String sqlSufix = ") as tmp GROUP BY fid "; - String sqlCountFilter = - (countFilter == null || countFilter.isEmpty()) - ? "" - : "HAVING SUM(count) " + countFilter; - String sqlRe = sqlPrefix + sqlUnion + sqlSufix + sqlCountFilter; - sqlCreate = "CREATE TABLE " + newTable + "(fid INTEGER NOT NULL, count INTEGER NOT NULL);"; - try { - adapter.execute(sqlCreate); - } catch (MetaStoreException e) { - LOG.error("Cannot create table " + newTable, e); - } - sqlFinal = "INSERT INTO " + newTable + " SELECT * FROM (" + sqlRe + ") temp;"; - } - return sqlFinal; - } - - private List getAccessCountTablesDuringLast(long lastInterval) { - List tableNames = new ArrayList<>(); - if (ruleManager == null || ruleManager.getStatesManager() == null) { - return tableNames; - } - - List accTables = null; + sqlCreate = "CREATE TABLE " + newTable + "(fid INTEGER NOT NULL, count INTEGER NOT NULL);"; try { - accTables = ruleManager.getStatesManager().getTablesForLast(lastInterval); + adapter.execute(sqlCreate); } catch (MetaStoreException e) { - LOG.error("Rule " + executionCtx.getRuleId() + " get access info tables exception", e); - } - - if (LOG.isDebugEnabled()) { - LOG.debug("Rule " + executionCtx.getRuleId() + " got " + accTables.size() + " tables:"); - int idx = 1; - for (AccessCountTable t : accTables) { - LOG.debug( - idx + ". " + (t.isEphemeral() ? " [TABLE] " : " ") + t.getTableName() + " "); - } + LOG.error("Cannot create table " + newTable, e); } - - if (accTables == null || accTables.isEmpty()) { - return tableNames; - } - - for (AccessCountTable t : accTables) { - tableNames.add(t.getTableName()); - if (t.isEphemeral()) { - dynamicCleanups.push("DROP TABLE IF EXISTS " + t.getTableName() + ";"); - } - } - return tableNames; + String sqlCountFilter = + (countFilter == null || countFilter.isEmpty()) + ? "" + : " HAVING count(*) " + countFilter; + sqlFinal = "INSERT INTO " + newTable + " SELECT fid, count(*) AS count FROM file_access\n" + + "WHERE access_time >= " + startTime + " AND access_time <= " + endTime + + " GROUP BY fid" + sqlCountFilter + " ;"; + return sqlFinal; } @Override diff --git a/smart-engine/src/test/java/org/smartdata/server/engine/rule/TestRuleExecutor.java b/smart-engine/src/test/java/org/smartdata/server/engine/rule/TestRuleExecutor.java index af02c86987d..f88e7211aa6 100644 --- a/smart-engine/src/test/java/org/smartdata/server/engine/rule/TestRuleExecutor.java +++ b/smart-engine/src/test/java/org/smartdata/server/engine/rule/TestRuleExecutor.java @@ -17,16 +17,14 @@ */ package org.smartdata.server.engine.rule; -import org.junit.Assert; import org.junit.Before; import org.junit.Test; -import org.smartdata.metastore.SqliteTestDaoBase; +import org.smartdata.metastore.TestDaoBase; import org.smartdata.metastore.dao.MetaStoreHelper; -import java.util.ArrayList; -import java.util.List; +import static org.junit.Assert.assertTrue; -public class TestRuleExecutor extends SqliteTestDaoBase { +public class TestRuleExecutor extends TestDaoBase { private MetaStoreHelper metaStoreHelper; @Before @@ -38,31 +36,28 @@ public void initActionDao() { public void generateSQL() { String countFilter = ""; String newTable = "test"; - List tableNames = new ArrayList<>(); - tableNames.add("blank_access_count_info"); String sql; - /*sql = "CREATE TABLE actual as SELECT fid, SUM(count)" + - " as count FROM (SELECT * FROM blank_access_count_info " + - "UNION ALL SELECT * FROM blank_access_count_info " + - "UNION ALL SELECT * FROM blank_access_count_info) as tmp GROUP BY fid"; - metaStoreHelper.execute(sql); - metaStoreHelper.dropTable("actual");*/ - // Test single element - sql = RuleExecutor.generateSQL(tableNames, newTable, countFilter, metaStore); + long interval = 60000; + long currentTimeMillis = System.currentTimeMillis(); + sql = RuleExecutor.generateSQL(newTable, countFilter, metaStore, currentTimeMillis - interval, + currentTimeMillis); try { metaStoreHelper.execute(sql); + assertTrue(sql.contains("GROUP BY fid ;")); metaStoreHelper.dropTable(newTable); } catch (Exception e) { - Assert.assertTrue(false); + assertTrue(false); } - // Test multiple elements - tableNames.add("blank_access_count_info"); - sql = RuleExecutor.generateSQL(tableNames, newTable, countFilter, metaStore); + // Test with count filter + countFilter = "> 10"; + sql = RuleExecutor.generateSQL(newTable, countFilter, metaStore, currentTimeMillis - interval, + currentTimeMillis); try { metaStoreHelper.execute(sql); + assertTrue(sql.contains("GROUP BY fid HAVING count(*) > 10 ;")); metaStoreHelper.dropTable(newTable); } catch (Exception e) { - Assert.assertTrue(false); + assertTrue(false); } } } diff --git a/smart-hadoop-support/smart-hadoop/src/test/java/org/smartdata/hdfs/metric/fetcher/TestCachedListFetcher.java b/smart-hadoop-support/smart-hadoop/src/test/java/org/smartdata/hdfs/metric/fetcher/TestCachedListFetcher.java index 4d918d620e4..c25c86a6c3f 100644 --- a/smart-hadoop-support/smart-hadoop/src/test/java/org/smartdata/hdfs/metric/fetcher/TestCachedListFetcher.java +++ b/smart-hadoop-support/smart-hadoop/src/test/java/org/smartdata/hdfs/metric/fetcher/TestCachedListFetcher.java @@ -36,7 +36,7 @@ import org.smartdata.hdfs.action.CacheFileAction; import org.smartdata.hdfs.action.UncacheFileAction; import org.smartdata.hdfs.scheduler.CacheScheduler; -import org.smartdata.metastore.SqliteTestDaoBase; +import org.smartdata.metastore.TestDaoBase; import org.smartdata.model.CachedFileStatus; import org.smartdata.model.FileInfo; @@ -46,7 +46,7 @@ import java.util.Map; -public class TestCachedListFetcher extends SqliteTestDaoBase { +public class TestCachedListFetcher extends TestDaoBase { private long fid; diff --git a/smart-hadoop-support/smart-hadoop/src/test/java/org/smartdata/hdfs/metric/fetcher/TestInotifyFetcher.java b/smart-hadoop-support/smart-hadoop/src/test/java/org/smartdata/hdfs/metric/fetcher/TestInotifyFetcher.java index a39a0354954..b5cec71f3c4 100644 --- a/smart-hadoop-support/smart-hadoop/src/test/java/org/smartdata/hdfs/metric/fetcher/TestInotifyFetcher.java +++ b/smart-hadoop-support/smart-hadoop/src/test/java/org/smartdata/hdfs/metric/fetcher/TestInotifyFetcher.java @@ -38,7 +38,7 @@ import org.smartdata.hdfs.CompatibilityHelperLoader; import org.smartdata.hdfs.MiniClusterFactory; import org.smartdata.metastore.MetaStore; -import org.smartdata.metastore.SqliteTestDaoBase; +import org.smartdata.metastore.TestDaoBase; import java.io.IOException; import java.io.OutputStream; @@ -49,7 +49,7 @@ import java.util.concurrent.Callable; import java.util.concurrent.Executors; -public class TestInotifyFetcher extends SqliteTestDaoBase { +public class TestInotifyFetcher extends TestDaoBase { private static final int BLOCK_SIZE = 1024; private static class EventApplierForTest extends InotifyEventApplier { diff --git a/smart-integration/src/test/java/org/smartdata/integration/TestCmdletRestApi.java b/smart-integration/src/test/java/org/smartdata/integration/TestCmdletRestApi.java index 44c10b1b1be..6d60055facc 100644 --- a/smart-integration/src/test/java/org/smartdata/integration/TestCmdletRestApi.java +++ b/smart-integration/src/test/java/org/smartdata/integration/TestCmdletRestApi.java @@ -101,7 +101,7 @@ public void testDeleteCmdlet() { .execute(Response::andReturn), response -> response.getStatusCode() == HttpStatus.NOT_FOUND_404, Duration.ofMillis(100), - Duration.ofSeconds(1) + Duration.ofSeconds(5) ); } diff --git a/smart-metastore/src/main/java/org/smartdata/metastore/DBType.java b/smart-metastore/src/main/java/org/smartdata/metastore/DBType.java index da80c1c19cf..096c8f56562 100644 --- a/smart-metastore/src/main/java/org/smartdata/metastore/DBType.java +++ b/smart-metastore/src/main/java/org/smartdata/metastore/DBType.java @@ -21,7 +21,5 @@ * Type of database. */ public enum DBType { - SQLITE, - MYSQL, POSTGRES } diff --git a/smart-metastore/src/main/java/org/smartdata/metastore/MetaStore.java b/smart-metastore/src/main/java/org/smartdata/metastore/MetaStore.java index d414bd663e4..0bcd23b76a4 100644 --- a/smart-metastore/src/main/java/org/smartdata/metastore/MetaStore.java +++ b/smart-metastore/src/main/java/org/smartdata/metastore/MetaStore.java @@ -33,6 +33,8 @@ import org.smartdata.metastore.dao.CompressionFileDao; import org.smartdata.metastore.dao.DaoProvider; import org.smartdata.metastore.dao.ErasureCodingPolicyDao; +import org.smartdata.metastore.dao.FileAccessDao; +import org.smartdata.metastore.dao.FileAccessPartitionDao; import org.smartdata.metastore.dao.FileDiffDao; import org.smartdata.metastore.dao.FileInfoDao; import org.smartdata.metastore.dao.FileStateDao; @@ -45,11 +47,8 @@ import org.smartdata.metastore.dao.SystemInfoDao; import org.smartdata.metastore.dao.UserActivityDao; import org.smartdata.metastore.dao.WhitelistDao; -import org.smartdata.metastore.dao.accesscount.AccessCountEventDao; -import org.smartdata.metastore.dao.accesscount.AccessCountTableDao; import org.smartdata.metastore.db.DbSchemaManager; import org.smartdata.metastore.db.metadata.DbMetadataProvider; -import org.smartdata.metastore.model.AccessCountTable; import org.smartdata.metastore.model.AggregatedAccessCounts; import org.smartdata.metastore.transaction.TransactionRunner; import org.smartdata.metastore.utils.MetaStoreUtils; @@ -112,8 +111,8 @@ public class MetaStore implements CopyMetaService, private final CacheFileDao cacheFileDao; private final StorageDao storageDao; private final FileDiffDao fileDiffDao; - private final AccessCountTableDao accessCountTableDao; - private final AccessCountEventDao accessCountEventDao; + private final FileAccessDao fileAccessDao; + private final FileAccessPartitionDao fileAccessPartitionDao; private final MetaStoreHelper metaStoreHelper; private final ClusterConfigDao clusterConfigDao; private final GlobalConfigDao globalConfigDao; @@ -144,8 +143,6 @@ public MetaStore(DBPool pool, fileInfoDao = daoProvider.fileInfoDao(); cacheFileDao = daoProvider.cacheFileDao(); storageDao = daoProvider.storageDao(); - accessCountTableDao = daoProvider.accessCountDao(); - accessCountEventDao = daoProvider.accessCountEventDao(); fileDiffDao = daoProvider.fileDiffDao(); metaStoreHelper = new MetaStoreHelper(pool.getDataSource()); clusterConfigDao = daoProvider.clusterConfigDao(); @@ -160,6 +157,8 @@ public MetaStore(DBPool pool, ecDao = daoProvider.ecDao(); whitelistDao = daoProvider.whitelistDao(); userActivityDao = daoProvider.userActivityDao(); + fileAccessPartitionDao = daoProvider.fileAccessPartitionDao(); + fileAccessDao = daoProvider.fileAccessDao(); } public DbMetadataProvider dbMetadataProvider() { @@ -182,12 +181,8 @@ public RuleDao ruleDao() { return ruleDao; } - public AccessCountTableDao accessCountTableDao() { - return accessCountTableDao; - } - - public AccessCountEventDao accessCountEventDao() { - return accessCountEventDao; + public FileAccessDao accessCountEventDao() { + return fileAccessDao; } public CacheFileDao cacheFileDao() { @@ -198,6 +193,10 @@ public FileInfoDao fileInfoDao() { return fileInfoDao; } + public FileAccessPartitionDao fileAccessPartitionDao() { + return fileAccessPartitionDao; + } + public PlatformTransactionManager transactionManager() { return defaultTransactionRunner.getTransactionManager(); @@ -351,9 +350,9 @@ public Map getFileIDs(Collection paths) } /** - * @param srcFileId the fid of old file. + * @param srcFileId the fid of old file. * @param destFileId the fid of new file that will take over the access - * count of old file. + * count of old file. */ public void updateAccessCountTableFileIds(long srcFileId, long destFileId) throws MetaStoreException { @@ -362,11 +361,7 @@ public void updateAccessCountTableFileIds(long srcFileId, long destFileId) } try { - defaultTransactionRunner.inTransaction(() -> { - List accessCountTables = - accessCountTableDao.getAllSortedTables(); - accessCountEventDao.updateFileIds(accessCountTables, srcFileId, destFileId); - }); + fileAccessDao.updateFileIds(srcFileId, destFileId); } catch (Exception e) { throw new MetaStoreException(e); } diff --git a/smart-metastore/src/main/java/org/smartdata/metastore/dao/accesscount/AccessEventAggregator.java b/smart-metastore/src/main/java/org/smartdata/metastore/accesscount/AccessEventAggregator.java similarity index 95% rename from smart-metastore/src/main/java/org/smartdata/metastore/dao/accesscount/AccessEventAggregator.java rename to smart-metastore/src/main/java/org/smartdata/metastore/accesscount/AccessEventAggregator.java index 8212f7a691d..1d8445fedb8 100644 --- a/smart-metastore/src/main/java/org/smartdata/metastore/dao/accesscount/AccessEventAggregator.java +++ b/smart-metastore/src/main/java/org/smartdata/metastore/accesscount/AccessEventAggregator.java @@ -15,7 +15,7 @@ * See the License for the specific language governing permissions and * limitations under the License. */ -package org.smartdata.metastore.dao.accesscount; +package org.smartdata.metastore.accesscount; import org.smartdata.metrics.FileAccessEvent; diff --git a/smart-metastore/src/main/java/org/smartdata/metastore/accesscount/DbAccessEventAggregator.java b/smart-metastore/src/main/java/org/smartdata/metastore/accesscount/DbAccessEventAggregator.java new file mode 100644 index 00000000000..15903da4a9d --- /dev/null +++ b/smart-metastore/src/main/java/org/smartdata/metastore/accesscount/DbAccessEventAggregator.java @@ -0,0 +1,84 @@ +/** + * 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.smartdata.metastore.accesscount; + +import lombok.extern.slf4j.Slf4j; +import org.smartdata.metastore.accesscount.failover.AccessCountContext; +import org.smartdata.metastore.accesscount.failover.Failover; +import org.smartdata.metastore.dao.FileInfoDao; +import org.smartdata.metastore.model.AggregatedAccessCounts; +import org.smartdata.metrics.FileAccessEvent; + +import java.util.Collections; +import java.util.List; +import java.util.Map; +import java.util.Objects; +import java.util.stream.Collectors; + +@Slf4j +public class DbAccessEventAggregator implements AccessEventAggregator { + + private final FileInfoDao fileInfoDao; + private final FileAccessManager dbTableManager; + private final Failover accessCountFailover; + + public DbAccessEventAggregator(FileInfoDao fileInfoDao, + FileAccessManager dbTableManager, + Failover failover) { + this.fileInfoDao = fileInfoDao; + this.dbTableManager = dbTableManager; + this.accessCountFailover = failover; + } + + @Override + public void aggregate(List events) { + List fileAccessCounts = getAggregatedAccessCounts(events); + AccessCountContext accessCountContext = + new AccessCountContext(fileAccessCounts); + accessCountFailover.execute(ctx -> dbTableManager.save(accessCountContext.getAccessCounts()), + accessCountContext); + } + + private List getAggregatedAccessCounts(List events) { + List paths = + events.stream() + .map(FileAccessEvent::getPath) + .collect(Collectors.toList()); + final Map pathFids = getFileIdMap(paths); + return events.stream() + .map(e -> { + Long fileId = pathFids.get(e.getPath()); + if (fileId != null) { + return AggregatedAccessCounts.fromEvent(e).withFileId(fileId); + } else { + return null; + } + }) + .filter(Objects::nonNull) + .collect(Collectors.toList()); + } + + private Map getFileIdMap(List paths) { + try { + return fileInfoDao.getPathFids(paths); + } catch (Exception e) { + log.error("Error fetching file ids for paths {}", paths, e); + return Collections.emptyMap(); + } + } +} diff --git a/smart-metastore/src/main/java/org/smartdata/metastore/accesscount/FileAccessManager.java b/smart-metastore/src/main/java/org/smartdata/metastore/accesscount/FileAccessManager.java new file mode 100644 index 00000000000..e22daf49b7d --- /dev/null +++ b/smart-metastore/src/main/java/org/smartdata/metastore/accesscount/FileAccessManager.java @@ -0,0 +1,117 @@ +/** + * 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.smartdata.metastore.accesscount; + +import lombok.extern.slf4j.Slf4j; +import org.smartdata.metastore.MetaStoreException; +import org.smartdata.metastore.dao.CacheFileDao; +import org.smartdata.metastore.dao.FileAccessDao; +import org.smartdata.metastore.dao.Searchable; +import org.smartdata.metastore.model.AggregatedAccessCounts; +import org.smartdata.metastore.model.SearchResult; +import org.smartdata.metastore.queries.PageRequest; +import org.smartdata.metastore.queries.sort.FileAccessInfoSortField; +import org.smartdata.metastore.transaction.TransactionRunner; +import org.smartdata.model.FileAccessInfo; +import org.smartdata.model.request.FileAccessInfoSearchRequest; + +import java.util.Collection; +import java.util.List; +import java.util.Map; +import java.util.Objects; +import java.util.function.Function; +import java.util.stream.Collectors; + +@Slf4j +public class FileAccessManager implements + Searchable { + + private final TransactionRunner transactionRunner; + private final FileAccessDao fileAccessDao; + private final CacheFileDao cacheFileDao; + + public FileAccessManager( + TransactionRunner transactionRunner, + FileAccessDao fileAccessDao, + CacheFileDao cacheFileDao) { + this.fileAccessDao = fileAccessDao; + this.cacheFileDao = cacheFileDao; + this.transactionRunner = transactionRunner; + } + + @Override + public SearchResult search(FileAccessInfoSearchRequest searchRequest, + PageRequest pageRequest) { + return fileAccessDao.search(searchRequest, pageRequest); + } + + @Override + public List search(FileAccessInfoSearchRequest searchRequest) { + return fileAccessDao.search(searchRequest); + } + + public void save(Collection accessCounts) { + if (accessCounts.isEmpty()) { + return; + } + try { + transactionRunner.inTransaction(() -> { + insertFileAccesses(accessCounts); + updateCachedFilesInMetastore(getAggregatedAccessCounts(accessCounts)); + }); + } catch (MetaStoreException e) { + log.error("Failed to save access counts", e); + throw new RuntimeException(e); + } + } + + private void insertFileAccesses( + Collection accessCounts) throws MetaStoreException { + try { + fileAccessDao.insert(accessCounts); + log.debug("Inserted values {} to file access table", accessCounts); + } catch (Exception e) { + log.error("Error inserting file accesses {}", accessCounts, e); + throw new MetaStoreException(e); + } + } + + private void updateCachedFilesInMetastore(Collection accessCounts) + throws MetaStoreException { + try { + cacheFileDao.update(accessCounts); + } catch (Exception e) { + log.error("Error updating cached files {}", accessCounts, e); + throw new MetaStoreException(e); + } + } + + private Collection getAggregatedAccessCounts( + Collection accessCounts) { + Map aggregatedAccessCounts = + accessCounts.stream() + .collect(Collectors.toMap( + AggregatedAccessCounts::getFileId, + Function.identity(), + AggregatedAccessCounts::merge + )); + return aggregatedAccessCounts.values().stream() + .filter(Objects::nonNull) + .collect(Collectors.toList()); + } +} diff --git a/smart-metastore/src/test/java/org/smartdata/metastore/dao/accesscount/NoOpAccessCountTableHandler.java b/smart-metastore/src/main/java/org/smartdata/metastore/accesscount/failover/AccessCountContext.java similarity index 61% rename from smart-metastore/src/test/java/org/smartdata/metastore/dao/accesscount/NoOpAccessCountTableHandler.java rename to smart-metastore/src/main/java/org/smartdata/metastore/accesscount/failover/AccessCountContext.java index ce440f3532c..4a7520ee66d 100644 --- a/smart-metastore/src/test/java/org/smartdata/metastore/dao/accesscount/NoOpAccessCountTableHandler.java +++ b/smart-metastore/src/main/java/org/smartdata/metastore/accesscount/failover/AccessCountContext.java @@ -15,22 +15,18 @@ * See the License for the specific language governing permissions and * limitations under the License. */ -package org.smartdata.metastore.dao.accesscount; +package org.smartdata.metastore.accesscount.failover; -import org.smartdata.metastore.MetaStoreException; -import org.smartdata.metastore.model.AccessCountTable; +import lombok.EqualsAndHashCode; +import lombok.Getter; +import lombok.RequiredArgsConstructor; +import org.smartdata.metastore.model.AggregatedAccessCounts; -import java.util.List; +import java.util.Collection; -public class NoOpAccessCountTableHandler implements AccessCountTableHandler { - @Override - public void dropTable(AccessCountTable accessCountTable) throws MetaStoreException { - - } - - @Override - public void aggregate(AccessCountTable destinationTable, List tablesToAggregate) - throws MetaStoreException { - - } +@Getter +@EqualsAndHashCode +@RequiredArgsConstructor +public class AccessCountContext { + private final Collection accessCounts; } diff --git a/smart-metastore/src/main/java/org/smartdata/metastore/accesscount/failover/AccessCountFailoverFactory.java b/smart-metastore/src/main/java/org/smartdata/metastore/accesscount/failover/AccessCountFailoverFactory.java new file mode 100644 index 00000000000..e7197e7f32d --- /dev/null +++ b/smart-metastore/src/main/java/org/smartdata/metastore/accesscount/failover/AccessCountFailoverFactory.java @@ -0,0 +1,46 @@ +/** + * 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.smartdata.metastore.accesscount.failover; + +import lombok.RequiredArgsConstructor; +import org.smartdata.conf.SmartConf; +import org.smartdata.metastore.accesscount.failover.impl.RetryAccessCountFailover; + +import static org.smartdata.conf.SmartConfKeys.SMART_ACCESS_COUNT_AGGREGATOR_FAILOVER_KEY; +import static org.smartdata.conf.SmartConfKeys.SMART_ACCESS_COUNT_AGGREGATOR_FAILOVER_MAX_RETRIES_DEFAULT; +import static org.smartdata.conf.SmartConfKeys.SMART_ACCESS_COUNT_AGGREGATOR_FAILOVER_MAX_RETRIES_KEY; + +@RequiredArgsConstructor +public class AccessCountFailoverFactory { + private final SmartConf conf; + + public Failover create() { + Failover.Strategy failoverStrategy = + conf.getEnum(SMART_ACCESS_COUNT_AGGREGATOR_FAILOVER_KEY, Failover.Strategy.FAIL); + switch (failoverStrategy) { + case FAIL: + return new Failover() { + }; + case SAVE_FAILED_WITH_RETRY: + default: + int maxRetries = conf.getInt(SMART_ACCESS_COUNT_AGGREGATOR_FAILOVER_MAX_RETRIES_KEY, + SMART_ACCESS_COUNT_AGGREGATOR_FAILOVER_MAX_RETRIES_DEFAULT); + return new RetryAccessCountFailover(maxRetries); + } + } +} diff --git a/smart-metastore/src/main/java/org/smartdata/metastore/dao/accesscount/AccessCountTableDao.java b/smart-metastore/src/main/java/org/smartdata/metastore/accesscount/failover/Failover.java similarity index 67% rename from smart-metastore/src/main/java/org/smartdata/metastore/dao/accesscount/AccessCountTableDao.java rename to smart-metastore/src/main/java/org/smartdata/metastore/accesscount/failover/Failover.java index 27188807ca6..4dd400b0a9e 100644 --- a/smart-metastore/src/main/java/org/smartdata/metastore/dao/accesscount/AccessCountTableDao.java +++ b/smart-metastore/src/main/java/org/smartdata/metastore/accesscount/failover/Failover.java @@ -15,20 +15,20 @@ * See the License for the specific language governing permissions and * limitations under the License. */ -package org.smartdata.metastore.dao.accesscount; +package org.smartdata.metastore.accesscount.failover; -import org.smartdata.metastore.model.AccessCountTable; +public interface Failover { -import java.util.List; + enum Strategy { + FAIL, + SAVE_FAILED_WITH_RETRY, + } -public interface AccessCountTableDao { - String TABLE_NAME = "access_count_table"; - - void insert(AccessCountTable accessCountTable); - - void delete(AccessCountTable table); - - List getAllSortedTables(); - - boolean tableExists(String name); + default void execute(Statement statement, T context) { + try { + statement.execute(context); + } catch (Exception e) { + throw new RuntimeException(e); + } + } } diff --git a/smart-metastore/src/main/java/org/smartdata/metastore/accesscount/failover/Statement.java b/smart-metastore/src/main/java/org/smartdata/metastore/accesscount/failover/Statement.java new file mode 100644 index 00000000000..4bb842a801f --- /dev/null +++ b/smart-metastore/src/main/java/org/smartdata/metastore/accesscount/failover/Statement.java @@ -0,0 +1,22 @@ +/** + * 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.smartdata.metastore.accesscount.failover; + +public interface Statement { + void execute(T context); +} diff --git a/smart-metastore/src/main/java/org/smartdata/metastore/accesscount/failover/impl/RetryAccessCountFailover.java b/smart-metastore/src/main/java/org/smartdata/metastore/accesscount/failover/impl/RetryAccessCountFailover.java new file mode 100644 index 00000000000..a941930b7ab --- /dev/null +++ b/smart-metastore/src/main/java/org/smartdata/metastore/accesscount/failover/impl/RetryAccessCountFailover.java @@ -0,0 +1,60 @@ +/** + * 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.smartdata.metastore.accesscount.failover.impl; + +import lombok.RequiredArgsConstructor; +import org.smartdata.metastore.accesscount.failover.AccessCountContext; +import org.smartdata.metastore.accesscount.failover.Failover; +import org.smartdata.metastore.accesscount.failover.Statement; +import org.smartdata.metastore.model.AggregatedAccessCounts; + +import java.util.ArrayList; +import java.util.List; + +@RequiredArgsConstructor +public class RetryAccessCountFailover implements Failover { + private final List failedAccessCounts = new ArrayList<>(); + private final int maxRetries; + private int retryCount = 0; + + @Override + public void execute(Statement statement, AccessCountContext context) { + try { + if (!failedAccessCounts.isEmpty()) { + List accessCounts = new ArrayList<>(context.getAccessCounts()); + accessCounts.addAll(failedAccessCounts); + context = new AccessCountContext(accessCounts); + } + statement.execute(context); + resetRetries(); + } catch (Exception e) { + retryCount++; + if (retryCount < maxRetries + 1) { + failedAccessCounts.addAll(context.getAccessCounts()); + } else { + resetRetries(); + throw new RuntimeException(e); + } + } + } + + private void resetRetries() { + retryCount = 0; + failedAccessCounts.clear(); + } +} diff --git a/smart-metastore/src/main/java/org/smartdata/metastore/dao/CacheFileDao.java b/smart-metastore/src/main/java/org/smartdata/metastore/dao/CacheFileDao.java index 8569da60788..258fdda5537 100644 --- a/smart-metastore/src/main/java/org/smartdata/metastore/dao/CacheFileDao.java +++ b/smart-metastore/src/main/java/org/smartdata/metastore/dao/CacheFileDao.java @@ -41,7 +41,7 @@ void insert(long fid, String path, long fromTime, void insert(List cachedFileStatusList); - int update(Long fid, Long lastAccessTime, Integer numAccessed); + int update(Long fid, Long lastAccessTime, long numAccessed); void update(Collection events); diff --git a/smart-metastore/src/main/java/org/smartdata/metastore/dao/DaoProvider.java b/smart-metastore/src/main/java/org/smartdata/metastore/dao/DaoProvider.java index 62debe833d5..d9e2ea04a1d 100644 --- a/smart-metastore/src/main/java/org/smartdata/metastore/dao/DaoProvider.java +++ b/smart-metastore/src/main/java/org/smartdata/metastore/dao/DaoProvider.java @@ -17,9 +17,6 @@ */ package org.smartdata.metastore.dao; -import org.smartdata.metastore.dao.accesscount.AccessCountEventDao; -import org.smartdata.metastore.dao.accesscount.AccessCountTableDao; - public interface DaoProvider { RuleDao ruleDao(); @@ -35,9 +32,7 @@ public interface DaoProvider { FileDiffDao fileDiffDao(); - AccessCountTableDao accessCountDao(); - - AccessCountEventDao accessCountEventDao(); + FileAccessDao fileAccessDao(); ClusterConfigDao clusterConfigDao(); @@ -64,4 +59,6 @@ public interface DaoProvider { StoragePolicyDao storagePolicyDao(); UserActivityDao userActivityDao(); + + FileAccessPartitionDao fileAccessPartitionDao(); } diff --git a/smart-metastore/src/main/java/org/smartdata/metastore/dao/DaoProviderFactory.java b/smart-metastore/src/main/java/org/smartdata/metastore/dao/DaoProviderFactory.java index e8f8a085c73..63156300b2b 100644 --- a/smart-metastore/src/main/java/org/smartdata/metastore/dao/DaoProviderFactory.java +++ b/smart-metastore/src/main/java/org/smartdata/metastore/dao/DaoProviderFactory.java @@ -19,9 +19,7 @@ import org.smartdata.metastore.DBPool; import org.smartdata.metastore.DBType; -import org.smartdata.metastore.dao.impl.DefaultDaoProvider; import org.smartdata.metastore.dao.postgres.PostgresDaoProvider; -import org.smartdata.metastore.dao.sqlite.SqliteDaoProvider; import org.springframework.transaction.PlatformTransactionManager; public class DaoProviderFactory { @@ -29,11 +27,8 @@ public DaoProvider createDaoProvider( DBPool dbPool, PlatformTransactionManager transactionManager, DBType dbType) { switch (dbType) { case POSTGRES: - return new PostgresDaoProvider(dbPool, transactionManager); - case SQLITE: - return new SqliteDaoProvider(dbPool, transactionManager); default: - return new DefaultDaoProvider(dbPool, transactionManager); + return new PostgresDaoProvider(dbPool, transactionManager); } } } diff --git a/smart-metastore/src/main/java/org/smartdata/metastore/dao/accesscount/AccessCountEventDao.java b/smart-metastore/src/main/java/org/smartdata/metastore/dao/FileAccessDao.java similarity index 54% rename from smart-metastore/src/main/java/org/smartdata/metastore/dao/accesscount/AccessCountEventDao.java rename to smart-metastore/src/main/java/org/smartdata/metastore/dao/FileAccessDao.java index 98e20c55e65..0a3ae756701 100644 --- a/smart-metastore/src/main/java/org/smartdata/metastore/dao/accesscount/AccessCountEventDao.java +++ b/smart-metastore/src/main/java/org/smartdata/metastore/dao/FileAccessDao.java @@ -15,47 +15,26 @@ * See the License for the specific language governing permissions and * limitations under the License. */ -package org.smartdata.metastore.dao.accesscount; +package org.smartdata.metastore.dao; import org.smartdata.metastore.MetaStoreException; -import org.smartdata.metastore.dao.Searchable; -import org.smartdata.metastore.model.AccessCountTable; import org.smartdata.metastore.model.AggregatedAccessCounts; import org.smartdata.metastore.queries.sort.FileAccessInfoSortField; import org.smartdata.model.FileAccessInfo; import org.smartdata.model.request.FileAccessInfoSearchRequest; import java.util.Collection; -import java.util.List; -import java.util.Set; -import java.util.stream.Collectors; -public interface AccessCountEventDao extends +public interface FileAccessDao extends Searchable { + String TABLE_NAME = "file_access"; String FILE_ID_FIELD = "fid"; String ACCESS_COUNT_FIELD = "count"; - String LAST_ACCESSED_TIME_FIELD = "last_accessed_time"; + String ACCESS_TIME_FIELD = "access_time"; - void insert( - AccessCountTable table, - Collection aggregatedAccessCounts) throws MetaStoreException; + void insert(Collection aggregatedAccessCounts) throws MetaStoreException; - void validate(AccessCountTable table) throws MetaStoreException; + void updateFileIds(long fidSrc, long fidDest) throws MetaStoreException; - void updateFileIds(List accessCountTables, - long fidSrc, long fidDest) throws MetaStoreException; - - static String unionTablesQuery(List tables) { - return tables.stream() - .map(AccessCountTable::getTableName) - .collect(Collectors.joining( - " UNION ALL SELECT * FROM ", "SELECT * FROM ", "")); - } - - static String unionTablesQuery(Set tables) { - return tables.stream() - .collect(Collectors.joining( - " UNION ALL SELECT * FROM ", "SELECT * FROM ", "")); - } } diff --git a/smart-metastore/src/main/java/org/smartdata/metastore/dao/accesscount/AccessCountTableHandler.java b/smart-metastore/src/main/java/org/smartdata/metastore/dao/FileAccessPartitionDao.java similarity index 70% rename from smart-metastore/src/main/java/org/smartdata/metastore/dao/accesscount/AccessCountTableHandler.java rename to smart-metastore/src/main/java/org/smartdata/metastore/dao/FileAccessPartitionDao.java index 7465cb745e8..0c392f5629f 100644 --- a/smart-metastore/src/main/java/org/smartdata/metastore/dao/accesscount/AccessCountTableHandler.java +++ b/smart-metastore/src/main/java/org/smartdata/metastore/dao/FileAccessPartitionDao.java @@ -15,16 +15,19 @@ * See the License for the specific language governing permissions and * limitations under the License. */ -package org.smartdata.metastore.dao.accesscount; +package org.smartdata.metastore.dao; import org.smartdata.metastore.MetaStoreException; -import org.smartdata.metastore.model.AccessCountTable; +import org.smartdata.metastore.model.FileAccessPartition; +import java.time.LocalDateTime; import java.util.List; -public interface AccessCountTableHandler { - void dropTable(AccessCountTable accessCountTable) throws MetaStoreException; +public interface FileAccessPartitionDao { - void aggregate(AccessCountTable destinationTable, - List tablesToAggregate) throws MetaStoreException; + void create(LocalDateTime date) throws MetaStoreException; + + List getAll(); + + void remove(FileAccessPartition partition); } diff --git a/smart-metastore/src/main/java/org/smartdata/metastore/dao/accesscount/AccessCountEventAggregatorFailover.java b/smart-metastore/src/main/java/org/smartdata/metastore/dao/accesscount/AccessCountEventAggregatorFailover.java deleted file mode 100644 index aa56403e0b2..00000000000 --- a/smart-metastore/src/main/java/org/smartdata/metastore/dao/accesscount/AccessCountEventAggregatorFailover.java +++ /dev/null @@ -1,77 +0,0 @@ -/** - * 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.smartdata.metastore.dao.accesscount; - -import org.smartdata.metastore.model.AggregatedAccessCounts; - -import java.util.Map; - -public interface AccessCountEventAggregatorFailover { - enum Strategy { - FAIL, - DROP_EVENTS, - SUBMIT_FAILED_EVENTS_LATER, - SUBMIT_NEW_FAILED_EVENTS_LATER - } - - /** - * Returns the map of {@link AggregatedAccessCounts} to be saved - * for subsequent submission attempts. - */ - Map handleError( - Map accessCounts, - Map previousUnmergedAccessCounts, - Exception error - ); - - static AccessCountEventAggregatorFailover fail() { - return (accessCounts, previousUnmergedAccessCounts, error) -> { - throw new RuntimeException(error); - }; - } - - static AccessCountEventAggregatorFailover dropEvents() { - return (accessCounts, previousUnmergedAccessCounts, error) -> - previousUnmergedAccessCounts; - } - - static AccessCountEventAggregatorFailover submitFailedEventsLater() { - return (accessCounts, previousUnmergedAccessCounts, error) -> - accessCounts; - } - - static AccessCountEventAggregatorFailover submitNewFailedEventsLater() { - return (accessCounts, previousUnmergedAccessCounts, error) -> { - accessCounts.keySet().removeAll(previousUnmergedAccessCounts.keySet()); - return accessCounts; - }; - } - - static AccessCountEventAggregatorFailover of(Strategy strategy) { - switch (strategy) { - case FAIL: - return fail(); - case SUBMIT_FAILED_EVENTS_LATER: - return submitFailedEventsLater(); - case SUBMIT_NEW_FAILED_EVENTS_LATER: - return submitNewFailedEventsLater(); - default: - return dropEvents(); - } - } -} diff --git a/smart-metastore/src/main/java/org/smartdata/metastore/dao/accesscount/AccessCountTableDeque.java b/smart-metastore/src/main/java/org/smartdata/metastore/dao/accesscount/AccessCountTableDeque.java deleted file mode 100644 index 68ee435872a..00000000000 --- a/smart-metastore/src/main/java/org/smartdata/metastore/dao/accesscount/AccessCountTableDeque.java +++ /dev/null @@ -1,85 +0,0 @@ -/** - * 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.smartdata.metastore.dao.accesscount; - -import org.smartdata.metastore.model.AccessCountTable; - -import java.util.ArrayList; -import java.util.List; -import java.util.Optional; -import java.util.concurrent.CompletableFuture; -import java.util.concurrent.ConcurrentLinkedDeque; - -import static com.google.common.base.Preconditions.checkNotNull; - -/** - * Use deque to accelerate remove operation. - */ -public class AccessCountTableDeque extends ConcurrentLinkedDeque { - private final TableAddOpListener listener; - private final AccessCountTableEvictor tableEvictor; - - public AccessCountTableDeque(AccessCountTableEvictor tableEvictor) { - this(tableEvictor, TableAddOpListener.noOp()); - } - - public AccessCountTableDeque(AccessCountTableEvictor tableEvictor, TableAddOpListener listener) { - super(); - this.listener = checkNotNull( - listener, "listener should not be null"); - this.tableEvictor = checkNotNull( - tableEvictor, "tableEvictor should not be null"); - } - - public CompletableFuture addAndNotifyListener(AccessCountTable table) { - boolean containsOverlappingTable = Optional.ofNullable(peekLast()) - .map(AccessCountTable::getEndTime) - .filter(endTime -> table.getEndTime() <= endTime) - .isPresent(); - - if (containsOverlappingTable) { - throw new IllegalArgumentException("Overlapping access count table: " + table); - } - - add(table); - return notifyListener(table); - } - - public CompletableFuture notifyListener(AccessCountTable table) { - return listener.tableAdded(this, table) - .thenAccept(this::evictTablesIfHigherGrainedCreated); - } - - private void evictTablesIfHigherGrainedCreated(AccessCountTable higherGrainedTable) { - if (higherGrainedTable == null) { - return; - } - - tableEvictor.evictTables(this, higherGrainedTable.getEndTime()); - } - - public List getTables(Long start, Long end) { - List results = new ArrayList<>(); - for (AccessCountTable table : this) { - if (table.getStartTime() >= start && table.getEndTime() <= end) { - results.add(table); - } - } - return results; - } -} diff --git a/smart-metastore/src/main/java/org/smartdata/metastore/dao/accesscount/AccessCountTableManager.java b/smart-metastore/src/main/java/org/smartdata/metastore/dao/accesscount/AccessCountTableManager.java deleted file mode 100644 index 0ee3272b4ea..00000000000 --- a/smart-metastore/src/main/java/org/smartdata/metastore/dao/accesscount/AccessCountTableManager.java +++ /dev/null @@ -1,371 +0,0 @@ -/** - * 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.smartdata.metastore.dao.accesscount; - -import lombok.Getter; -import org.apache.hadoop.conf.Configuration; -import org.slf4j.Logger; -import org.slf4j.LoggerFactory; -import org.smartdata.conf.SmartConfKeys; -import org.smartdata.metastore.MetaStore; -import org.smartdata.metastore.MetaStoreException; -import org.smartdata.metastore.dao.Searchable; -import org.smartdata.metastore.model.AccessCountTable; -import org.smartdata.metastore.model.AggregatedAccessCounts; -import org.smartdata.metastore.model.SearchResult; -import org.smartdata.metastore.queries.PageRequest; -import org.smartdata.metastore.queries.sort.FileAccessInfoSortField; -import org.smartdata.metastore.transaction.TransactionRunner; -import org.smartdata.metastore.utils.TimeGranularity; -import org.smartdata.model.FileAccessInfo; -import org.smartdata.model.TimeInterval; -import org.smartdata.model.request.FileAccessInfoSearchRequest; -import org.smartdata.utils.DateTimeUtils; - -import java.util.ArrayList; -import java.util.Collection; -import java.util.Collections; -import java.util.Deque; -import java.util.List; -import java.util.Optional; -import java.util.Set; -import java.util.concurrent.ExecutorService; -import java.util.stream.Collectors; - -import static org.smartdata.conf.SmartConfKeys.SMART_ACCESS_COUNT_AGGREGATION_INTERVAL_MS; -import static org.smartdata.conf.SmartConfKeys.SMART_ACCESS_COUNT_AGGREGATION_INTERVAL_MS_DEFAULT; -import static org.smartdata.conf.SmartConfKeys.SMART_ACCESS_COUNT_AGGREGATOR_FAILOVER_KEY; -import static org.smartdata.metastore.utils.TimeGranularity.decreaseGranularity; -import static org.smartdata.metastore.utils.TimeGranularity.increaseGranularity; -import static org.springframework.transaction.annotation.Isolation.SERIALIZABLE; - -@Getter -public class AccessCountTableManager implements - Searchable { - private final InMemoryAccessEventAggregator accessEventAggregator; - private final InMemoryAccessCountTableManager inMemoryTableManager; - private final DbAccessCountTableManager dbTableManager; - private final TransactionRunner transactionRunner; - private final int defaultHotFilesLimit; - - public static final Logger LOG = - LoggerFactory.getLogger(AccessCountTableManager.class); - - public AccessCountTableManager(MetaStore metaStore, - ExecutorService service, - Configuration configuration) { - int aggregationIntervalMs = configuration.getInt( - SMART_ACCESS_COUNT_AGGREGATION_INTERVAL_MS, - SMART_ACCESS_COUNT_AGGREGATION_INTERVAL_MS_DEFAULT); - AccessCountEventAggregatorFailover.Strategy eventAggregatorFailoverStrategy = - configuration.getEnum(SMART_ACCESS_COUNT_AGGREGATOR_FAILOVER_KEY, - AccessCountEventAggregatorFailover.Strategy.SUBMIT_NEW_FAILED_EVENTS_LATER); - - this.accessEventAggregator = new InMemoryAccessEventAggregator( - metaStore.fileInfoDao(), - this::onAggregationWindowFinish, - AccessCountEventAggregatorFailover.of(eventAggregatorFailoverStrategy), - aggregationIntervalMs); - - this.transactionRunner = new TransactionRunner(metaStore.transactionManager()); - transactionRunner.setIsolationLevel(SERIALIZABLE); - - this.dbTableManager = new DbAccessCountTableManager( - metaStore.getDataSource(), - transactionRunner, - metaStore.accessCountTableDao(), - metaStore.accessCountEventDao(), - metaStore.cacheFileDao()); - this.inMemoryTableManager = new InMemoryAccessCountTableManager( - dbTableManager, service, configuration); - this.defaultHotFilesLimit = configuration.getInt( - SmartConfKeys.SMART_TOP_HOT_FILES_NUM_KEY, - SmartConfKeys.SMART_TOP_HOT_FILES_NUM_DEFAULT); - - recoverTables(); - } - - public List getTablesForLast(long intervalMillis) throws MetaStoreException { - Deque secondDeque = - inMemoryTableManager.getTablesOfGranularity(TimeGranularity.SECOND); - if (secondDeque.isEmpty()) { - return Collections.emptyList(); - } - - long endTime = secondDeque.getLast().getEndTime(); - return getAccessCountTables(endTime - intervalMillis, endTime); - } - - public void createTable(AccessCountTable table) throws MetaStoreException { - try { - transactionRunner.inTransaction(() -> { - dbTableManager.createTable(table); - inMemoryTableManager.addTable(table); - }); - } catch (MetaStoreException exception) { - LOG.error("Error creating access count table {}", table, exception); - throw exception; - } - } - - private void onAggregationWindowFinish( - long windowStart, - long windowEnd, - Collection aggregatedAccessCounts) { - if (aggregatedAccessCounts.isEmpty()) { - return; - } - - AccessCountTable table = new AccessCountTable(windowStart, windowEnd); - try { - transactionRunner.inTransaction(() -> { - createTable(table); - dbTableManager.handleAggregatedEvents(table, aggregatedAccessCounts); - }); - } catch (MetaStoreException exception) { - LOG.error("Error creating access count table {}", table, exception); - throw new RuntimeException(exception); - } - } - - private void recoverTables() { - try { - transactionRunner.inTransaction(() -> { - List tables = dbTableManager.getTables(); - inMemoryTableManager.recoverTables(tables); - }); - } catch (MetaStoreException exception) { - LOG.error("Error recovering existing access count tables", exception); - } - } - - /** - * Getting access count tables by interval: - * 1. found base granularity by searching interval, if tables are not exist, increase granularity - * 2. get access count tables by base granularity with startTime less or equals searching endTime - * 3. Filter tables by starTime less or equals searching endTime. - * 4. Filter tables by endTime > searching startTime and if parentTable != null, - * also filter by endTime <= parentTable.endTime. It is needed because we are getting - * all tables of some granularity. - * 5. For all found tables from the list we should add corresponding tables to the result - * according to their intervals and search granularity. For non-second granularity we get child - * tables recursively. If we have case when one of the searching border (startTime or endTime) - * bigger (less) than table startTime (endTime), we have to create partial table, which is based - * on current table (in some cases we create partial table base on parentTable, - * when prentTable != null and there are no tables of small granularity between - * parentTable.startTime and table.startTime or between searching startTime and - * parentTable.endTime). Add corresponding tables to the result and update searching startTime and - * 6. Decrease searching granularity and update it, check that searching - * startTime = endTime, if no, repeat from p.2 - * - * @param startTime start time of the search interval - * @param endTime end time of the search interval - * @return List of access count tables - * @throws MetaStoreException error - */ - private List getAccessCountTables(long startTime, - long endTime) throws MetaStoreException { - final TimeGranularity startGranularity = TimeGranularity.of(endTime - startTime); - TimeGranularity searchIntervalGranularity = startGranularity; - Collection tables = inMemoryTableManager.getTablesOfGranularity( - searchIntervalGranularity).stream() - .filter(t -> t.getStartTime() <= endTime) - .collect(Collectors.toList()); - boolean foundTables = !tables.isEmpty(); - while (!foundTables) { - searchIntervalGranularity = increaseGranularity(searchIntervalGranularity); - if (searchIntervalGranularity == null) { - searchIntervalGranularity = startGranularity; - foundTables = true; - } else { - tables = inMemoryTableManager.getTablesOfGranularity( - searchIntervalGranularity); - Long accessCountTableStartTime = tables.stream() - .findFirst() - .map(AccessCountTable::getStartTime) - .orElse(Long.MAX_VALUE); - foundTables = accessCountTableStartTime <= startTime; - } - } - return getAccessCountTables(startTime, endTime, null, searchIntervalGranularity); - } - - private List getAccessCountTables(long startTime, - long endTime, - AccessCountTable parentTable, - TimeGranularity baseGranularity) - throws MetaStoreException { - final TimeGranularity startGranularity = TimeGranularity.of(endTime - startTime); - final List result = new ArrayList<>(); - TimeGranularity searchIntervalGranularity; - if (baseGranularity != null) { - searchIntervalGranularity = baseGranularity; - } else { - searchIntervalGranularity = startGranularity; - } - do { - long finalStartTime = startTime; - Collection tables = inMemoryTableManager.getTablesOfGranularity( - searchIntervalGranularity).stream() - .filter(t -> t.getStartTime() <= endTime) - .filter(t -> { - if (parentTable != null) { - //include child tables which has endTime after startTime - // and includes into parent table - return t.getEndTime() > finalStartTime && t.getEndTime() <= parentTable.getEndTime(); - } - return t.getEndTime() > finalStartTime; - }) - .collect(Collectors.toList()); - int n = 0; - for (AccessCountTable table : tables) { - n++; - //skip tables if it starts after specified startTime - if (table.getStartTime() > endTime) { - continue; - } - if (table.getStartTime() <= startTime) { - if (table.getStartTime() == startTime) { - if (table.getEndTime() <= endTime) { - result.add(table); - startTime = table.getEndTime(); - } else { - //table.endTime > endTime - result.addAll(getChildOrCreatePartialTables(searchIntervalGranularity, - table, startTime, endTime)); - startTime = endTime; - } - } else { - if (table.getEndTime() <= endTime) { - result.addAll(getChildOrCreatePartialTables(searchIntervalGranularity, - table, startTime, table.getEndTime())); - startTime = table.getEndTime(); - } else { - result.addAll(getChildOrCreatePartialTables(searchIntervalGranularity, - table, startTime, endTime)); - startTime = endTime; - } - } - continue; - } - //table.startTime > startTime - if (table.getEndTime() <= endTime) { - //if table is child and the first, we have to create partial table for this interval - if (parentTable != null && n <= 1) { - //create partial table based on parent table from startTime to table.startTime - addPartialTable(parentTable, startTime, table.getStartTime()) - .ifPresent(result::add); - } - result.add(table); - startTime = table.getEndTime(); - } else { - addPartialTable(table, table.getStartTime(), endTime) - .ifPresent(result::add); - startTime = endTime; - } - } - searchIntervalGranularity = decreaseGranularity(searchIntervalGranularity); - if (searchIntervalGranularity == null && startTime != endTime) { - if (parentTable != null) { - //create partial table based on parent table from startTime to endTime - addPartialTable(parentTable, startTime, endTime) - .ifPresent(result::add); - } - startTime = endTime; - } - } while (startTime != endTime); - return result; - } - - private List getChildOrCreatePartialTables( - TimeGranularity searchIntervalGranularity, - AccessCountTable table, - long startTime, - long endTime) - throws MetaStoreException { - if (searchIntervalGranularity != TimeGranularity.SECOND) { - return getAccessCountTables(startTime, endTime, table, null); - } else { - return addPartialTable(table, startTime, endTime) - .map(Collections::singletonList) - .orElse(Collections.emptyList()); - } - } - - private Optional addPartialTable(AccessCountTable sourceTable, - long startTime, - long endTime) throws MetaStoreException { - // The access count table interval intersects with the search interval - if (!inMemoryTableManager.tableExists(startTime, endTime)) { - // Create ephemeral table with file access events whose last access time - // is greater than or equal to the startTime. We also assume that file accesses - // occurred evenly over time, so we can simply divide the number of file accesses - // during the table interval by the ratio of the search interval and the table interval - AccessCountTable partialTable = - new AccessCountTable(startTime, endTime, true); - dbTableManager.createPartialTable(partialTable, sourceTable); - return Optional.of(partialTable); - } else { - return Optional.empty(); - } - } - - @Override - public SearchResult search(FileAccessInfoSearchRequest searchRequest, - PageRequest pageRequest) { - try { - return transactionRunner.inTransaction(() -> { - Set tables = getAccessCountTables(searchRequest.getLastAccessedTime()); - if (tables.isEmpty()) { - return SearchResult.of(Collections.emptyList(), 0); - } - return dbTableManager.getFileAccessInfoList(searchRequest.withAccessCountTables(tables), - pageRequest); - }); - } catch (MetaStoreException e) { - LOG.error("Failed to get file access count information with pagination", e); - throw new RuntimeException(e); - } - } - - @Override - public List search(FileAccessInfoSearchRequest searchRequest) { - try { - return transactionRunner.inTransaction(() -> { - Set tables = getAccessCountTables(searchRequest.getLastAccessedTime()); - if (tables.isEmpty()) { - return Collections.emptyList(); - } - return dbTableManager.getFileAccessInfoList(searchRequest.withAccessCountTables(tables)); - }); - } catch (MetaStoreException e) { - LOG.error("Failed to get file access count information", e); - throw new RuntimeException(e); - } - } - - private Set getAccessCountTables(TimeInterval timeInterval) throws MetaStoreException { - long startTime = Optional.ofNullable(DateTimeUtils.intervalStartToEpoch(timeInterval)) - .orElse(0L); - long endTime = Optional.ofNullable(DateTimeUtils.intervalEndToEpoch(timeInterval)) - .orElse(System.currentTimeMillis()); - return getAccessCountTables(startTime, endTime).stream() - .map(AccessCountTable::getTableName) - .collect(Collectors.toSet()); - } -} diff --git a/smart-metastore/src/main/java/org/smartdata/metastore/dao/accesscount/AggregatingTableAddOpListener.java b/smart-metastore/src/main/java/org/smartdata/metastore/dao/accesscount/AggregatingTableAddOpListener.java deleted file mode 100644 index 7d41f867618..00000000000 --- a/smart-metastore/src/main/java/org/smartdata/metastore/dao/accesscount/AggregatingTableAddOpListener.java +++ /dev/null @@ -1,100 +0,0 @@ -/** - * 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.smartdata.metastore.dao.accesscount; - -import org.slf4j.Logger; -import org.slf4j.LoggerFactory; -import org.smartdata.metastore.model.AccessCountTable; - -import java.util.List; -import java.util.Set; -import java.util.concurrent.CompletableFuture; -import java.util.concurrent.ConcurrentHashMap; -import java.util.concurrent.ExecutorService; - -public class AggregatingTableAddOpListener implements TableAddOpListener { - static final Logger LOG = LoggerFactory.getLogger(AggregatingTableAddOpListener.class); - - private final AccessCountTableDeque coarseGrainedTableDeque; - private final AccessCountTableHandler tableAggregator; - private final ExecutorService executorService; - private final Set tablesUnderConstruction; - private final long millisPerGranularity; - - AggregatingTableAddOpListener( - AccessCountTableDeque deque, - AccessCountTableHandler tableAggregator, - ExecutorService executorService, - long millisPerGranularity) { - this.coarseGrainedTableDeque = deque; - this.tableAggregator = tableAggregator; - this.executorService = executorService; - this.millisPerGranularity = millisPerGranularity; - tablesUnderConstruction = ConcurrentHashMap.newKeySet(); - } - - @Override - public CompletableFuture tableAdded( - AccessCountTableDeque fineGrainedTableDeque, AccessCountTable table) { - final AccessCountTable lastCoarseGrainedTable = lastCoarseGrainedTableFor(table.getEndTime()); - - // Todo: optimize contains - if (coarseGrainedTableDeque.contains(lastCoarseGrainedTable)) { - return CompletableFuture.completedFuture(lastCoarseGrainedTable); - } - - final List tablesToAggregate = - fineGrainedTableDeque.getTables( - lastCoarseGrainedTable.getStartTime(), lastCoarseGrainedTable.getEndTime()); - - if (!tablesToAggregate.isEmpty() - && !tablesUnderConstruction.contains(lastCoarseGrainedTable)) { - tablesUnderConstruction.add(lastCoarseGrainedTable); - return aggregateTableAsync(lastCoarseGrainedTable, tablesToAggregate); - } - - return TableAddOpListener.EMPTY_RESULT; - } - - private CompletableFuture aggregateTableAsync( - AccessCountTable lastCoarseGrainedTable, List tablesToAggregate) { - return CompletableFuture.supplyAsync( - () -> aggregateTable(lastCoarseGrainedTable, tablesToAggregate), executorService); - } - - private AccessCountTable aggregateTable( - AccessCountTable lastCoarseGrainedTable, List tablesToAggregate) { - try { - tableAggregator.aggregate(lastCoarseGrainedTable, tablesToAggregate); - coarseGrainedTableDeque.addAndNotifyListener(lastCoarseGrainedTable); - } catch (Exception e) { - LOG.error( - "Add AccessCount Table {} error", - lastCoarseGrainedTable.getTableName(), e); - } - tablesUnderConstruction.remove(lastCoarseGrainedTable); - return lastCoarseGrainedTable; - } - - protected AccessCountTable lastCoarseGrainedTableFor(long endTime) { - long lastEnd = endTime - (endTime % millisPerGranularity); - long lastStart = lastEnd - millisPerGranularity; - return new AccessCountTable(lastStart, lastEnd); - } - // Todo: WeekTableListener, MonthTableListener, YearTableListener -} diff --git a/smart-metastore/src/main/java/org/smartdata/metastore/dao/accesscount/CountTableEvictor.java b/smart-metastore/src/main/java/org/smartdata/metastore/dao/accesscount/CountTableEvictor.java deleted file mode 100644 index 77eb2c6f18e..00000000000 --- a/smart-metastore/src/main/java/org/smartdata/metastore/dao/accesscount/CountTableEvictor.java +++ /dev/null @@ -1,49 +0,0 @@ -/** - * 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.smartdata.metastore.dao.accesscount; - -import org.smartdata.metastore.model.AccessCountTable; - -import java.util.Iterator; - -public class CountTableEvictor extends AccessCountTableEvictor { - private final int maxCount; - - public CountTableEvictor(AccessCountTableHandler tableDeleter, int count) { - super(tableDeleter); - this.maxCount = count; - } - - @Override - public void evictTables(AccessCountTableDeque tables, long lastAggregatedIntervalEndTimestamp) { - int elementsToRemove = tables.size() - maxCount; - - for (Iterator iterator = tables.iterator(); - iterator.hasNext() && elementsToRemove-- > 0; ) { - AccessCountTable table = iterator.next(); - - if (table.getEndTime() > lastAggregatedIntervalEndTimestamp) { - // table belongs to not yet aggregated higher granularity interval - return; - } - - iterator.remove(); - dropTable(table); - } - } -} diff --git a/smart-metastore/src/main/java/org/smartdata/metastore/dao/accesscount/DbAccessCountTableManager.java b/smart-metastore/src/main/java/org/smartdata/metastore/dao/accesscount/DbAccessCountTableManager.java deleted file mode 100644 index 904f2949f7c..00000000000 --- a/smart-metastore/src/main/java/org/smartdata/metastore/dao/accesscount/DbAccessCountTableManager.java +++ /dev/null @@ -1,209 +0,0 @@ -/** - * 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.smartdata.metastore.dao.accesscount; - -import org.slf4j.Logger; -import org.slf4j.LoggerFactory; -import org.smartdata.metastore.MetaStore; -import org.smartdata.metastore.MetaStoreException; -import org.smartdata.metastore.dao.AbstractDao; -import org.smartdata.metastore.dao.CacheFileDao; -import org.smartdata.metastore.model.AccessCountTable; -import org.smartdata.metastore.model.AggregatedAccessCounts; -import org.smartdata.metastore.model.SearchResult; -import org.smartdata.metastore.queries.PageRequest; -import org.smartdata.metastore.queries.sort.FileAccessInfoSortField; -import org.smartdata.metastore.transaction.TransactionRunner; -import org.smartdata.model.FileAccessInfo; -import org.smartdata.model.request.FileAccessInfoSearchRequest; - -import javax.sql.DataSource; - -import java.util.Collection; -import java.util.List; -import java.util.stream.Collectors; - -public class DbAccessCountTableManager - extends AbstractDao implements AccessCountTableHandler { - static final Logger LOG = LoggerFactory.getLogger(DbAccessCountTableManager.class); - - private final TransactionRunner transactionRunner; - private final AccessCountTableDao accessCountTableDao; - private final AccessCountEventDao accessCountEventDao; - private final CacheFileDao cacheFileDao; - - public DbAccessCountTableManager(MetaStore metastore) { - this( - metastore.getDataSource(), - new TransactionRunner(metastore.transactionManager()), - metastore.accessCountTableDao(), - metastore.accessCountEventDao(), - metastore.cacheFileDao() - ); - } - - public DbAccessCountTableManager( - DataSource dataSource, - TransactionRunner transactionRunner, - AccessCountTableDao accessCountTableDao, - AccessCountEventDao accessCountEventDao, - CacheFileDao cacheFileDao) { - super(dataSource, AccessCountTableDao.TABLE_NAME); - this.accessCountTableDao = accessCountTableDao; - this.accessCountEventDao = accessCountEventDao; - this.cacheFileDao = cacheFileDao; - this.transactionRunner = transactionRunner; - } - - @Override - public void aggregate( - AccessCountTable destinationTable, - List tablesToAggregate) throws MetaStoreException { - if (tablesToAggregate.isEmpty()) { - return; - } - - transactionRunner.inTransaction(() -> { - createTable(destinationTable); - aggregateTablesInternal(destinationTable, tablesToAggregate); - }); - } - - public void createTable(AccessCountTable table) throws MetaStoreException { - transactionRunner.inTransaction(() -> { - createAccessCountTable(table); - accessCountTableDao.insert(table); - }); - } - - public void createPartialTable(AccessCountTable dest, AccessCountTable source) - throws MetaStoreException { - transactionRunner.inTransaction(() -> { - createAccessCountTable(dest); - fillPartialTable(dest, source); - }); - } - - public SearchResult getFileAccessInfoList( - FileAccessInfoSearchRequest searchRequest, - PageRequest pageRequest) { - return accessCountEventDao.search(searchRequest, pageRequest); - } - - public List getFileAccessInfoList(FileAccessInfoSearchRequest searchRequest) { - return accessCountEventDao.search(searchRequest); - } - - @Override - public void dropTable(AccessCountTable accessCountTable) throws MetaStoreException { - transactionRunner.inTransaction(() -> { - dropDbTable(accessCountTable); - accessCountTableDao.delete(accessCountTable); - }); - } - - public void handleAggregatedEvents( - AccessCountTable table, - Collection accessCounts) throws MetaStoreException { - if (accessCounts.isEmpty()) { - return; - } - - transactionRunner.inTransaction(() -> { - insertAccessCountsToMetastore(table, accessCounts); - updateCachedFilesInMetastore(accessCounts); - }); - } - - public List getTables() { - return accessCountTableDao.getAllSortedTables() - .stream() - .filter(this::isValid) - .collect(Collectors.toList()); - } - - private boolean isValid(AccessCountTable table) { - try { - accessCountEventDao.validate(table); - return true; - } catch (MetaStoreException exception) { - LOG.error("Can't recover table {}, dropping it", table, exception); - dropDbTable(table); - return false; - } - } - - private void fillPartialTable(AccessCountTable dest, AccessCountTable source) { - String statement = - "INSERT INTO " + dest.getTableName() - + " SELECT src.fid, ROUND(src.count * " + dest.intervalRatio(source) - + "), src.last_accessed_time" - + " FROM " + source.getTableName() + " as src" - + " WHERE src.last_accessed_time >= " + source.getStartTime(); - - jdbcTemplate.execute(statement); - } - - private void aggregateTablesInternal(AccessCountTable destination, - List sources) { - String query = "INSERT INTO " + destination.getTableName() - + " SELECT aggregated_events.fid, " - + "aggregated_events.count, " - + "aggregated_events.last_accessed_time " - + "FROM (" - + "SELECT fid, SUM(count) AS count, MAX(last_accessed_time) as last_accessed_time " - + "FROM (" + AccessCountEventDao.unionTablesQuery(sources) + ") as union_events " - + "GROUP BY fid) AS aggregated_events " - + "JOIN file ON file.fid = aggregated_events.fid;"; - - LOG.debug("Executing access count tables aggregation: {}", query); - jdbcTemplate.execute(query); - } - - private void createAccessCountTable(AccessCountTable table) { - String createStatement = "CREATE TABLE IF NOT EXISTS " + table.getTableName() + "(" - + "fid BIGINT NOT NULL, " - + "count INTEGER NOT NULL, " - + "last_accessed_time BIGINT NOT NULL)"; - - jdbcTemplate.execute(createStatement); - } - - private void dropDbTable(AccessCountTable accessCountTable) { - String sql = "DROP TABLE IF EXISTS " + accessCountTable.getTableName(); - jdbcTemplate.execute(sql); - } - - private void insertAccessCountsToMetastore( - AccessCountTable table, Collection accessCounts) { - try { - accessCountEventDao.insert(table, accessCounts); - LOG.debug("Inserted values {} to access count table {}", accessCounts, table); - } catch (Exception e) { - LOG.error("Error inserting access counts {} to table {}", accessCounts, table, e); - } - } - - private void updateCachedFilesInMetastore(Collection accessCounts) { - try { - cacheFileDao.update(accessCounts); - } catch (Exception e) { - LOG.error("Error updating cached files {}", accessCounts, e); - } - } -} diff --git a/smart-metastore/src/main/java/org/smartdata/metastore/dao/accesscount/InMemoryAccessCountTableManager.java b/smart-metastore/src/main/java/org/smartdata/metastore/dao/accesscount/InMemoryAccessCountTableManager.java deleted file mode 100644 index 123703e8137..00000000000 --- a/smart-metastore/src/main/java/org/smartdata/metastore/dao/accesscount/InMemoryAccessCountTableManager.java +++ /dev/null @@ -1,195 +0,0 @@ -/** - * 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.smartdata.metastore.dao.accesscount; - -import org.apache.hadoop.conf.Configuration; -import org.slf4j.Logger; -import org.slf4j.LoggerFactory; -import org.smartdata.metastore.model.AccessCountTable; -import org.smartdata.metastore.utils.TimeGranularity; - -import java.util.ArrayDeque; -import java.util.Deque; -import java.util.HashMap; -import java.util.List; -import java.util.Map; -import java.util.Optional; -import java.util.concurrent.ExecutorService; - -import static org.smartdata.conf.SmartConfKeys.SMART_ACCESS_COUNT_AGGREGATION_INTERVAL_MS; -import static org.smartdata.conf.SmartConfKeys.SMART_ACCESS_COUNT_AGGREGATION_INTERVAL_MS_DEFAULT; -import static org.smartdata.conf.SmartConfKeys.SMART_NUM_DAY_TABLES_TO_KEEP_DEFAULT; -import static org.smartdata.conf.SmartConfKeys.SMART_NUM_DAY_TABLES_TO_KEEP_KEY; -import static org.smartdata.conf.SmartConfKeys.SMART_NUM_HOUR_TABLES_TO_KEEP_DEFAULT; -import static org.smartdata.conf.SmartConfKeys.SMART_NUM_HOUR_TABLES_TO_KEEP_KEY; -import static org.smartdata.conf.SmartConfKeys.SMART_NUM_HOUR_TABLES_TO_KEEP_MIN; -import static org.smartdata.conf.SmartConfKeys.SMART_NUM_MINUTE_TABLES_TO_KEEP_DEFAULT; -import static org.smartdata.conf.SmartConfKeys.SMART_NUM_MINUTE_TABLES_TO_KEEP_KEY; -import static org.smartdata.conf.SmartConfKeys.SMART_NUM_MINUTE_TABLES_TO_KEEP_MIN; -import static org.smartdata.conf.SmartConfKeys.SMART_NUM_SECOND_TABLES_TO_KEEP_DEFAULT; -import static org.smartdata.conf.SmartConfKeys.SMART_NUM_SECOND_TABLES_TO_KEEP_KEY; -import static org.smartdata.metastore.utils.Constants.ONE_MINUTE_IN_MILLIS; - -public class InMemoryAccessCountTableManager { - private final Map tableDeques; - private final Configuration configuration; - private AccessCountTableDeque secondTableDeque; - - public static final Logger LOG = - LoggerFactory.getLogger(InMemoryAccessCountTableManager.class); - - public InMemoryAccessCountTableManager( - AccessCountTableHandler tableHandler, - ExecutorService executorService, - Configuration configuration) { - this.configuration = configuration; - this.tableDeques = new HashMap<>(); - - initTables(tableHandler, executorService); - } - - public void recoverTables(List tables) { - if (tables.isEmpty()) { - LOG.info("No existing access count tables to recover."); - return; - } - - LOG.info("Loading existing access count tables: {}", tables); - for (AccessCountTable table : tables) { - if (tableDeques.containsKey(table.getGranularity())) { - tableDeques.get(table.getGranularity()).add(table); - } - } - - // aggregate old tables if needed - AccessCountTable lastOldTable = tables.get(tables.size() - 1); - Optional.ofNullable(tableDeques.get(lastOldTable.getGranularity())) - .ifPresent(deque -> deque.notifyListener(lastOldTable)); - - LOG.info("Existing access count tables were successfully loaded"); - } - - public void addTable(AccessCountTable accessCountTable) { - if (LOG.isDebugEnabled()) { - LOG.debug(accessCountTable.toString()); - } - secondTableDeque.addAndNotifyListener(accessCountTable); - } - - public Deque getTablesOfGranularity(TimeGranularity timeGranularity) { - return Optional.>ofNullable( - tableDeques.get(timeGranularity)) - .orElseGet(ArrayDeque::new); - } - - public boolean tableExists(long start, long end) { - TimeGranularity granularity = TimeGranularity.of(end - start); - AccessCountTable fakeTable = new AccessCountTable(start, end); - return tableDeques.containsKey(granularity) - && tableDeques.get(granularity).contains(fakeTable); - } - - private void initTables( - AccessCountTableHandler tableHandler, - ExecutorService executorService) { - - TableAddOpListener dayTableListener = - createPerDayTableListener(tableHandler, executorService); - - TableAddOpListener hourTableListener = - createPerHourTableListener(tableHandler, executorService, dayTableListener); - - TableAddOpListener minuteTableListener = - createPerMinuteTableListener(tableHandler, executorService, hourTableListener); - - createSecondTableDeque(tableHandler, minuteTableListener); - } - - private TableAddOpListener createPerDayTableListener( - AccessCountTableHandler tableHandler, ExecutorService executorService) { - int perDayAccessTablesCount = configuration.getInt( - SMART_NUM_DAY_TABLES_TO_KEEP_KEY, - SMART_NUM_DAY_TABLES_TO_KEEP_DEFAULT); - AccessCountTableDeque dayTableDeque = new AccessCountTableDeque( - new CountTableEvictor(tableHandler, perDayAccessTablesCount)); - - tableDeques.put(TimeGranularity.DAY, dayTableDeque); - return TableAddOpListener.perDay(dayTableDeque, tableHandler, executorService); - } - - private TableAddOpListener createPerHourTableListener( - AccessCountTableHandler tableHandler, - ExecutorService executorService, - TableAddOpListener dayTableListener) { - int perHourAccessTablesCount = getAccessTablesCount( - SMART_NUM_HOUR_TABLES_TO_KEEP_KEY, - SMART_NUM_HOUR_TABLES_TO_KEEP_DEFAULT, - SMART_NUM_HOUR_TABLES_TO_KEEP_MIN); - AccessCountTableDeque hourTableDeque = new AccessCountTableDeque( - new CountTableEvictor(tableHandler, perHourAccessTablesCount), dayTableListener); - - tableDeques.put(TimeGranularity.HOUR, hourTableDeque); - return TableAddOpListener.perHour(hourTableDeque, tableHandler, executorService); - } - - private TableAddOpListener createPerMinuteTableListener( - AccessCountTableHandler tableHandler, - ExecutorService executorService, - TableAddOpListener hourTableListener) { - int perMinuteAccessTablesCount = getAccessTablesCount( - SMART_NUM_MINUTE_TABLES_TO_KEEP_KEY, - SMART_NUM_MINUTE_TABLES_TO_KEEP_DEFAULT, - SMART_NUM_MINUTE_TABLES_TO_KEEP_MIN); - AccessCountTableDeque minuteTableDeque = new AccessCountTableDeque( - new CountTableEvictor(tableHandler, perMinuteAccessTablesCount), hourTableListener); - - tableDeques.put(TimeGranularity.MINUTE, minuteTableDeque); - return TableAddOpListener.perMinute(minuteTableDeque, tableHandler, executorService); - } - - private void createSecondTableDeque( - AccessCountTableHandler tableHandler, - TableAddOpListener minuteTableListener) { - int aggregationIntervalMs = configuration.getInt( - SMART_ACCESS_COUNT_AGGREGATION_INTERVAL_MS, - SMART_ACCESS_COUNT_AGGREGATION_INTERVAL_MS_DEFAULT); - int minimalSecondAccessTablesCount = - (int) (ONE_MINUTE_IN_MILLIS / aggregationIntervalMs); - - int perSecondAccessTablesCount = getAccessTablesCount( - SMART_NUM_SECOND_TABLES_TO_KEEP_KEY, - SMART_NUM_SECOND_TABLES_TO_KEEP_DEFAULT, - minimalSecondAccessTablesCount); - this.secondTableDeque = new AccessCountTableDeque( - new CountTableEvictor(tableHandler, perSecondAccessTablesCount), minuteTableListener); - - tableDeques.put(TimeGranularity.SECOND, secondTableDeque); - } - - private int getAccessTablesCount(String configKey, int defaultValue, int minimalCount) { - int tableCount = configuration.getInt(configKey, defaultValue); - - if (tableCount < minimalCount) { - String errorMessage = String.format( - "Wrong value for option %s. It should be at least %d", configKey, minimalCount); - LOG.error(errorMessage); - throw new IllegalArgumentException(errorMessage); - } - return tableCount; - } -} diff --git a/smart-metastore/src/main/java/org/smartdata/metastore/dao/accesscount/InMemoryAccessEventAggregator.java b/smart-metastore/src/main/java/org/smartdata/metastore/dao/accesscount/InMemoryAccessEventAggregator.java deleted file mode 100644 index b1b82cf46eb..00000000000 --- a/smart-metastore/src/main/java/org/smartdata/metastore/dao/accesscount/InMemoryAccessEventAggregator.java +++ /dev/null @@ -1,166 +0,0 @@ -/** - * 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.smartdata.metastore.dao.accesscount; - -import lombok.Data; -import org.apache.commons.lang.StringUtils; -import org.slf4j.Logger; -import org.slf4j.LoggerFactory; -import org.smartdata.metastore.dao.FileInfoDao; -import org.smartdata.metastore.model.AggregatedAccessCounts; -import org.smartdata.metrics.FileAccessEvent; - -import java.util.ArrayList; -import java.util.Collection; -import java.util.Collections; -import java.util.HashMap; -import java.util.HashSet; -import java.util.List; -import java.util.Map; -import java.util.Set; -import java.util.stream.Collectors; - -public class InMemoryAccessEventAggregator implements AccessEventAggregator { - public static final Logger LOG = LoggerFactory.getLogger(InMemoryAccessEventAggregator.class); - - private final FileInfoDao fileInfoDao; - private final WindowClosedCallback windowClosedCallback; - private final AccessCountEventAggregatorFailover failover; - private final List eventBuffer; - private final long aggregationGranularity; - private Window currentWindow; - private Map unmergedAccessCounts = new HashMap<>(); - - public InMemoryAccessEventAggregator( - FileInfoDao fileInfoDao, - WindowClosedCallback windowClosedCallback, - AccessCountEventAggregatorFailover failover, - long aggregationGranularity) { - this.fileInfoDao = fileInfoDao; - this.windowClosedCallback = windowClosedCallback; - this.aggregationGranularity = aggregationGranularity; - this.failover = failover; - this.eventBuffer = new ArrayList<>(); - } - - @Override - public void aggregate(List events) { - if (currentWindow == null && !events.isEmpty()) { - currentWindow = assignWindow(events.get(0).getTimestamp()); - } - for (FileAccessEvent event : events) { - if (!currentWindow.contains(event.getTimestamp())) { - // New Window occurs - closeCurrentWindow(); - currentWindow = assignWindow(event.getTimestamp()); - eventBuffer.clear(); - } - // Exclude watermark event - if (!StringUtils.isBlank(event.getPath())) { - eventBuffer.add(event); - } - } - } - - private void closeCurrentWindow() { - Map aggregatedAccessCounts = aggregateWindowEvents(); - - try { - windowClosedCallback.onWindowClosed( - currentWindow.start, currentWindow.end, aggregatedAccessCounts.values()); - } catch (Exception exception) { - unmergedAccessCounts = failover.handleError( - aggregatedAccessCounts, unmergedAccessCounts, exception); - } - } - - private Map aggregateWindowEvents() { - if (eventBuffer.isEmpty() && unmergedAccessCounts.isEmpty()) { - return Collections.emptyMap(); - } - - Map allAccessesCounts = aggregateAccessCounts(eventBuffer); - Set newAccessPaths = new HashSet<>(allAccessesCounts.keySet()); - mergeMapsInPlace(allAccessesCounts, unmergedAccessCounts); - - Map pathToIds = getFileIds(allAccessesCounts); - - unmergedAccessCounts = new HashMap<>(); - Map accessCountsWithFileId = new HashMap<>(); - - allAccessesCounts.forEach((path, accessesCounts) -> { - Long fileId = pathToIds.get(path); - - if (fileId != null) { - accessCountsWithFileId.put(path, accessesCounts.withFileId(fileId)); - return; - } - - if (newAccessPaths.contains(path)) { - // save only files from event buffer, i.e. drop unsuccessful files - // from previous unmergedAccessCounts - unmergedAccessCounts.put(path, accessesCounts); - } - }); - return accessCountsWithFileId; - } - - private Map getFileIds(Map allAccessesCounts) { - try { - return fileInfoDao.getPathFids(allAccessesCounts.keySet()); - } catch (Exception exception) { - LOG.error("Error fetching file ids for paths {}", allAccessesCounts.keySet(), exception); - return Collections.emptyMap(); - } - } - - private void mergeMapsInPlace(Map resultMap, - Map mapToMerge) { - mapToMerge.forEach((key, value) -> resultMap.merge(key, value, AggregatedAccessCounts::merge)); - } - - private Map aggregateAccessCounts(List events) { - return events.stream() - .collect(Collectors.toMap( - FileAccessEvent::getPath, - AggregatedAccessCounts::fromEvent, - AggregatedAccessCounts::merge - )); - } - - private Window assignWindow(long time) { - long start = time - (time % aggregationGranularity); - return new Window(start, start + aggregationGranularity); - } - - @Data - public static class Window { - private final long start; - private final long end; - - // [start, end) - public boolean contains(long time) { - return this.start <= time && this.end > time; - } - } - - public interface WindowClosedCallback { - void onWindowClosed(long windowStart, long windowEnd, - Collection aggregatedAccessCounts); - } -} diff --git a/smart-metastore/src/main/java/org/smartdata/metastore/dao/accesscount/TableAddOpListener.java b/smart-metastore/src/main/java/org/smartdata/metastore/dao/accesscount/TableAddOpListener.java deleted file mode 100644 index 157fedb183c..00000000000 --- a/smart-metastore/src/main/java/org/smartdata/metastore/dao/accesscount/TableAddOpListener.java +++ /dev/null @@ -1,59 +0,0 @@ -/** - * 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.smartdata.metastore.dao.accesscount; - -import org.smartdata.metastore.model.AccessCountTable; -import org.smartdata.metastore.utils.Constants; - -import java.util.concurrent.CompletableFuture; -import java.util.concurrent.ExecutorService; - -public interface TableAddOpListener { - CompletableFuture EMPTY_RESULT = CompletableFuture.completedFuture(null); - - CompletableFuture tableAdded( - AccessCountTableDeque fineGrainedTableDeque, AccessCountTable table); - - static TableAddOpListener perDay( - AccessCountTableDeque deque, - AccessCountTableHandler tableAggregator, - ExecutorService service) { - return new AggregatingTableAddOpListener( - deque, tableAggregator, service, Constants.ONE_DAY_IN_MILLIS); - } - - static TableAddOpListener perHour( - AccessCountTableDeque deque, - AccessCountTableHandler tableAggregator, - ExecutorService service) { - return new AggregatingTableAddOpListener( - deque, tableAggregator, service, Constants.ONE_HOUR_IN_MILLIS); - } - - static TableAddOpListener perMinute( - AccessCountTableDeque deque, - AccessCountTableHandler tableAggregator, - ExecutorService service) { - return new AggregatingTableAddOpListener( - deque, tableAggregator, service, Constants.ONE_MINUTE_IN_MILLIS); - } - - static TableAddOpListener noOp() { - return (deque, table) -> EMPTY_RESULT; - } -} diff --git a/smart-metastore/src/main/java/org/smartdata/metastore/dao/impl/DefaultAccessCountEventDao.java b/smart-metastore/src/main/java/org/smartdata/metastore/dao/impl/DefaultAccessCountEventDao.java deleted file mode 100644 index 1706ddda10d..00000000000 --- a/smart-metastore/src/main/java/org/smartdata/metastore/dao/impl/DefaultAccessCountEventDao.java +++ /dev/null @@ -1,151 +0,0 @@ -/** - * 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.smartdata.metastore.dao.impl; - -import com.google.common.collect.ImmutableMap; -import org.smartdata.metastore.MetaStoreException; -import org.smartdata.metastore.SearchableAbstractDao; -import org.smartdata.metastore.dao.FileInfoDao; -import org.smartdata.metastore.dao.accesscount.AccessCountEventDao; -import org.smartdata.metastore.model.AccessCountTable; -import org.smartdata.metastore.model.AggregatedAccessCounts; -import org.smartdata.metastore.queries.MetastoreQuery; -import org.smartdata.metastore.queries.MetastoreQueryExecutor; -import org.smartdata.metastore.queries.sort.FileAccessInfoSortField; -import org.smartdata.model.FileAccessInfo; -import org.smartdata.model.request.FileAccessInfoSearchRequest; -import org.springframework.jdbc.core.ColumnMapRowMapper; -import org.springframework.jdbc.core.simple.SimpleJdbcInsert; -import org.springframework.transaction.PlatformTransactionManager; - -import javax.sql.DataSource; - -import java.sql.ResultSet; -import java.sql.SQLException; -import java.util.Collection; -import java.util.HashMap; -import java.util.List; -import java.util.Map; - -import static org.smartdata.metastore.queries.MetastoreQuery.select; -import static org.smartdata.metastore.queries.expression.MetastoreQueryDsl.betweenEpochInclusive; -import static org.smartdata.metastore.queries.expression.MetastoreQueryDsl.in; -import static org.smartdata.metastore.queries.expression.MetastoreQueryDsl.like; - -public class DefaultAccessCountEventDao - extends - SearchableAbstractDao - implements AccessCountEventDao { - - private static final String TABLE_NAME = "access_count_table"; - private final MetastoreQueryExecutor queryExecutor; - - public DefaultAccessCountEventDao( - DataSource dataSource, PlatformTransactionManager transactionManager) { - super(dataSource, transactionManager, TABLE_NAME); - - this.queryExecutor = new MetastoreQueryExecutor(dataSource, transactionManager); - } - - @Override - public void insert(AccessCountTable table, - Collection aggregatedAccessCounts) { - insert(simpleJdbcInsert(table), aggregatedAccessCounts, this::toMap); - } - - protected Map toMap(AggregatedAccessCounts accessCounts) { - Map parameters = new HashMap<>(); - parameters.put(AccessCountEventDao.FILE_ID_FIELD, accessCounts.getFileId()); - parameters.put(AccessCountEventDao.ACCESS_COUNT_FIELD, accessCounts.getAccessCount()); - parameters.put(AccessCountEventDao.LAST_ACCESSED_TIME_FIELD, - accessCounts.getLastAccessedTimestamp()); - return parameters; - } - - @Override - public void validate(AccessCountTable table) throws MetaStoreException { - MetastoreQuery query = select(FILE_ID_FIELD, ACCESS_COUNT_FIELD, LAST_ACCESSED_TIME_FIELD) - .from(table.getTableName()) - .limit(1); - try { - queryExecutor.execute(query, new ColumnMapRowMapper()); - } catch (Exception exception) { - throw new MetaStoreException(exception); - } - } - - @Override - public void updateFileIds(List accessCountTables, - long srcFileId, long destFileId) throws MetaStoreException { - Exception lastException = null; - int failedNum = 0; - - for (AccessCountTable table : accessCountTables) { - String statement = "UPDATE " + table.getTableName() - + " SET fid = " + destFileId - + " WHERE fid = " + srcFileId; - - try { - jdbcTemplate.execute(statement); - } catch (Exception exception) { - lastException = exception; - failedNum++; - } - } - // Otherwise, ignore the exception because table evictor can evict access - // count tables, which are not synchronized. Even so, there is no impact on - // the measurement for data temperature. - if (failedNum == accessCountTables.size()) { - // Throw exception if all tables are not updated. - throw new MetaStoreException("Failed to update fid!", lastException); - } - } - - private SimpleJdbcInsert simpleJdbcInsert(AccessCountTable table) { - return new SimpleJdbcInsert(dataSource) - .withTableName(table.getTableName()); - } - - @Override - protected MetastoreQuery searchQuery(FileAccessInfoSearchRequest searchRequest) { - return select("access_counts.fid AS fid", - "SUM(access_counts.count) AS count", - "MAX(access_counts.last_accessed_time) as last_accessed_time", - "file.path as path") - .fromSubQuery(AccessCountEventDao.unionTablesQuery(searchRequest.getAccessCountTables()), - "access_counts") - .join("file", ImmutableMap.of("access_counts.fid", "file.fid")) - .where( - in("access_counts.fid", searchRequest.getIds()), - like("file.path", searchRequest.getPathLike()), - betweenEpochInclusive("access_counts.last_accessed_time", - searchRequest.getLastAccessedTime()) - ) - .groupBy("access_counts.fid", "file.path"); - } - - @Override - protected FileAccessInfo mapRow(ResultSet rs, int rowNum) throws SQLException { - return FileAccessInfo.builder() - .setFid(rs.getLong(AccessCountEventDao.FILE_ID_FIELD)) - .setPath(rs.getString(FileInfoDao.FILE_PATH_FIELD)) - .setAccessCount(rs.getInt(AccessCountEventDao.ACCESS_COUNT_FIELD)) - .setLastAccessedTime(rs.getLong(AccessCountEventDao.LAST_ACCESSED_TIME_FIELD)) - .build(); - } -} diff --git a/smart-metastore/src/main/java/org/smartdata/metastore/dao/impl/DefaultAccessCountTableDao.java b/smart-metastore/src/main/java/org/smartdata/metastore/dao/impl/DefaultAccessCountTableDao.java deleted file mode 100644 index 04d0296b737..00000000000 --- a/smart-metastore/src/main/java/org/smartdata/metastore/dao/impl/DefaultAccessCountTableDao.java +++ /dev/null @@ -1,97 +0,0 @@ -/** - * 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.smartdata.metastore.dao.impl; - -import org.smartdata.metastore.dao.AbstractDao; -import org.smartdata.metastore.dao.accesscount.AccessCountTableDao; -import org.smartdata.metastore.model.AccessCountTable; -import org.smartdata.metastore.queries.MetastoreQuery; -import org.smartdata.metastore.queries.MetastoreQueryExecutor; -import org.springframework.transaction.PlatformTransactionManager; - -import javax.sql.DataSource; - -import java.sql.ResultSet; -import java.sql.SQLException; -import java.util.HashMap; -import java.util.List; -import java.util.Map; - -import static org.smartdata.metastore.queries.MetastoreQuery.select; -import static org.smartdata.metastore.queries.MetastoreQuery.selectAll; -import static org.smartdata.metastore.queries.expression.MetastoreQueryDsl.equal; -import static org.smartdata.metastore.queries.sort.Sorting.ascending; - -public class DefaultAccessCountTableDao extends AbstractDao implements AccessCountTableDao { - - protected final MetastoreQueryExecutor queryExecutor; - - public DefaultAccessCountTableDao( - DataSource dataSource, PlatformTransactionManager transactionManager) { - super(dataSource, TABLE_NAME); - - this.queryExecutor = new MetastoreQueryExecutor(dataSource, transactionManager); - } - - @Override - public void insert(AccessCountTable table) { - if (!tableExists(table.getTableName())) { - insert(table, this::toMap); - } - } - - @Override - public boolean tableExists(String name) { - MetastoreQuery query = select("1") - .from(TABLE_NAME) - .where( - equal("table_name", name) - ) - .limit(1L); - - return queryExecutor.executeCount(query) != 0; - } - - @Override - public void delete(AccessCountTable table) { - final String sql = "DELETE FROM access_count_table WHERE table_name = ?"; - jdbcTemplate.update(sql, table.getTableName()); - } - - @Override - public List getAllSortedTables() { - MetastoreQuery query = selectAll() - .from(TABLE_NAME) - .orderBy(ascending("start_time")); - return queryExecutor.execute(query, this::mapRow); - } - - private Map toMap(AccessCountTable accessCountTable) { - Map parameters = new HashMap<>(); - parameters.put("table_name", accessCountTable.getTableName()); - parameters.put("start_time", accessCountTable.getStartTime()); - parameters.put("end_time", accessCountTable.getEndTime()); - return parameters; - } - - private AccessCountTable mapRow(ResultSet resultSet, int i) throws SQLException { - return new AccessCountTable( - resultSet.getLong("start_time"), - resultSet.getLong("end_time")); - } -} diff --git a/smart-metastore/src/main/java/org/smartdata/metastore/dao/impl/DefaultCacheFileDao.java b/smart-metastore/src/main/java/org/smartdata/metastore/dao/impl/DefaultCacheFileDao.java index c2286575030..9dcbfe7bf0d 100644 --- a/smart-metastore/src/main/java/org/smartdata/metastore/dao/impl/DefaultCacheFileDao.java +++ b/smart-metastore/src/main/java/org/smartdata/metastore/dao/impl/DefaultCacheFileDao.java @@ -97,7 +97,7 @@ public void insert(List cachedFileStatusList) { } @Override - public int update(Long fid, Long lastAccessTime, Integer numAccessed) { + public int update(Long fid, Long lastAccessTime, long numAccessed) { String sql = "UPDATE cached_file SET last_access_time = ?, accessed_num = ? WHERE fid = ?"; return jdbcTemplate.update(sql, lastAccessTime, numAccessed, fid); } @@ -136,8 +136,7 @@ private void merge( long lastAccessTime = Math.max( cachedFileStatus.getLastAccessTime(), aggregatedAccessCounts.getLastAccessedTimestamp()); - - int accessCounts = cachedFileStatus.getNumAccessed() + long accessCounts = cachedFileStatus.getNumAccessed() + aggregatedAccessCounts.getAccessCount(); update(aggregatedAccessCounts.getFileId(), lastAccessTime, accessCounts); diff --git a/smart-metastore/src/main/java/org/smartdata/metastore/dao/impl/DefaultDaoProvider.java b/smart-metastore/src/main/java/org/smartdata/metastore/dao/impl/DefaultDaoProvider.java index 18ea0331628..3f3c7632f5f 100644 --- a/smart-metastore/src/main/java/org/smartdata/metastore/dao/impl/DefaultDaoProvider.java +++ b/smart-metastore/src/main/java/org/smartdata/metastore/dao/impl/DefaultDaoProvider.java @@ -27,6 +27,8 @@ import org.smartdata.metastore.dao.CompressionFileDao; import org.smartdata.metastore.dao.DaoProvider; import org.smartdata.metastore.dao.ErasureCodingPolicyDao; +import org.smartdata.metastore.dao.FileAccessDao; +import org.smartdata.metastore.dao.FileAccessPartitionDao; import org.smartdata.metastore.dao.FileDiffDao; import org.smartdata.metastore.dao.FileInfoDao; import org.smartdata.metastore.dao.FileStateDao; @@ -39,8 +41,6 @@ import org.smartdata.metastore.dao.SystemInfoDao; import org.smartdata.metastore.dao.UserActivityDao; import org.smartdata.metastore.dao.WhitelistDao; -import org.smartdata.metastore.dao.accesscount.AccessCountEventDao; -import org.smartdata.metastore.dao.accesscount.AccessCountTableDao; import org.springframework.transaction.PlatformTransactionManager; import javax.sql.DataSource; @@ -90,13 +90,8 @@ public FileDiffDao fileDiffDao() { } @Override - public AccessCountTableDao accessCountDao() { - return new DefaultAccessCountTableDao(dataSource, transactionManager); - } - - @Override - public AccessCountEventDao accessCountEventDao() { - return new DefaultAccessCountEventDao(dataSource, transactionManager); + public FileAccessDao fileAccessDao() { + return new DefaultFileAccessDao(dataSource, transactionManager); } @Override @@ -163,4 +158,9 @@ public StoragePolicyDao storagePolicyDao() { public UserActivityDao userActivityDao() { return new DefaultUserActivityDao(dataSource, transactionManager); } + + @Override + public FileAccessPartitionDao fileAccessPartitionDao() { + return new DefaultFileAccessPartitionDao(dataSource); + } } diff --git a/smart-metastore/src/main/java/org/smartdata/metastore/dao/impl/DefaultFileAccessDao.java b/smart-metastore/src/main/java/org/smartdata/metastore/dao/impl/DefaultFileAccessDao.java new file mode 100644 index 00000000000..e1b7d169020 --- /dev/null +++ b/smart-metastore/src/main/java/org/smartdata/metastore/dao/impl/DefaultFileAccessDao.java @@ -0,0 +1,109 @@ +/** + * 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.smartdata.metastore.dao.impl; + +import org.smartdata.metastore.MetaStoreException; +import org.smartdata.metastore.SearchableAbstractDao; +import org.smartdata.metastore.dao.FileAccessDao; +import org.smartdata.metastore.dao.FileInfoDao; +import org.smartdata.metastore.model.AggregatedAccessCounts; +import org.smartdata.metastore.queries.MetastoreQuery; +import org.smartdata.metastore.queries.sort.FileAccessInfoSortField; +import org.smartdata.model.FileAccessInfo; +import org.smartdata.model.request.FileAccessInfoSearchRequest; +import org.springframework.jdbc.core.simple.SimpleJdbcInsert; +import org.springframework.transaction.PlatformTransactionManager; + +import javax.sql.DataSource; + +import java.sql.ResultSet; +import java.sql.SQLException; +import java.util.Collection; +import java.util.HashMap; +import java.util.Map; + +import static org.smartdata.metastore.queries.MetastoreQuery.select; +import static org.smartdata.metastore.queries.expression.MetastoreQueryDsl.betweenEpochInclusive; +import static org.smartdata.metastore.queries.expression.MetastoreQueryDsl.in; +import static org.smartdata.metastore.queries.expression.MetastoreQueryDsl.like; + +public class DefaultFileAccessDao + extends + SearchableAbstractDao + implements FileAccessDao { + + public DefaultFileAccessDao( + DataSource dataSource, PlatformTransactionManager transactionManager) { + super(dataSource, transactionManager, TABLE_NAME); + } + + @Override + public void insert(Collection aggregatedAccessCounts) { + insert(new SimpleJdbcInsert(dataSource).withTableName(TABLE_NAME), aggregatedAccessCounts, + this::toMap); + } + + protected Map toMap(AggregatedAccessCounts accessCounts) { + Map parameters = new HashMap<>(); + parameters.put(FileAccessDao.FILE_ID_FIELD, accessCounts.getFileId()); + parameters.put(FileAccessDao.ACCESS_TIME_FIELD, + accessCounts.getLastAccessedTimestamp()); + return parameters; + } + + @Override + public void updateFileIds(long srcFileId, long destFileId) throws MetaStoreException { + String statement = "UPDATE " + TABLE_NAME + + " SET fid = " + destFileId + + " WHERE fid = " + srcFileId; + try { + jdbcTemplate.execute(statement); + } catch (Exception exception) { + throw new MetaStoreException("Failed to update fid!", exception); + } + } + + @Override + protected MetastoreQuery searchQuery(FileAccessInfoSearchRequest searchRequest) { + return select("fid", + "count", + "access_time", + "path") + .fromSubQuery("SELECT file.fid, count(*) AS count,\n" + + "MAX(file_access.access_time) as access_time, file.path as path\n" + + "FROM file_access\n" + + " JOIN file ON file_access.fid = file.fid\n" + + "GROUP BY file.fid, file.path", "f") + .where( + in("fid", searchRequest.getIds()), + like("path", searchRequest.getPathLike()), + betweenEpochInclusive("access_time", + searchRequest.getLastAccessedTime()) + ); + } + + @Override + protected FileAccessInfo mapRow(ResultSet rs, int rowNum) throws SQLException { + return FileAccessInfo.builder() + .setFid(rs.getLong(FileAccessDao.FILE_ID_FIELD)) + .setPath(rs.getString(FileInfoDao.FILE_PATH_FIELD)) + .setAccessCount(rs.getInt(FileAccessDao.ACCESS_COUNT_FIELD)) + .setLastAccessedTime(rs.getLong(FileAccessDao.ACCESS_TIME_FIELD)) + .build(); + } +} diff --git a/smart-metastore/src/main/java/org/smartdata/metastore/dao/impl/DefaultFileAccessPartitionDao.java b/smart-metastore/src/main/java/org/smartdata/metastore/dao/impl/DefaultFileAccessPartitionDao.java new file mode 100644 index 00000000000..b7bc37ea117 --- /dev/null +++ b/smart-metastore/src/main/java/org/smartdata/metastore/dao/impl/DefaultFileAccessPartitionDao.java @@ -0,0 +1,85 @@ +/** + * 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.smartdata.metastore.dao.impl; + +import lombok.extern.slf4j.Slf4j; +import org.smartdata.metastore.MetaStoreException; +import org.smartdata.metastore.dao.AbstractDao; +import org.smartdata.metastore.dao.FileAccessPartitionDao; +import org.smartdata.metastore.model.FileAccessPartition; +import org.springframework.dao.EmptyResultDataAccessException; + +import javax.sql.DataSource; + +import java.time.LocalDateTime; +import java.util.Collections; +import java.util.List; + +@Slf4j +public class DefaultFileAccessPartitionDao extends AbstractDao implements FileAccessPartitionDao { + + private static final String CREATE_NEW_PARTITION_ERR_MSG = + "Failed to create new partition for file_access table"; + + public DefaultFileAccessPartitionDao(DataSource dataSource) { + super(dataSource, ""); + } + + @Override + public void create(LocalDateTime date) throws MetaStoreException { + try { + Integer result = + jdbcTemplate.queryForObject("select create_file_access_partition(?);", Integer.class, + date); + if (result == null) { + throw new MetaStoreException(CREATE_NEW_PARTITION_ERR_MSG); + } + if (result == 1) { + log.info("Created partition for file_access table for date {}", date); + } + } catch (Exception e) { + throw new MetaStoreException(CREATE_NEW_PARTITION_ERR_MSG, e); + } + } + + @Override + public List getAll() { + String query = "SELECT inhrelid AS id, inhrelid::regclass AS name, " + + "cast(REPLACE(REPLACE(inhrelid::regclass::text, 'file_access_', ''),'_','-') as date) " + + "as partition_date FROM pg_catalog.pg_inherits " + + "WHERE inhparent = 'file_access'::regclass " + + "ORDER BY partition_date ASC;"; + try { + return jdbcTemplate.query(query, + (rs, rowNum) -> new FileAccessPartition(rs.getLong("id"), + rs.getString("name"), + rs.getDate("partition_date").toLocalDate())); + } catch (EmptyResultDataAccessException e) { + return Collections.emptyList(); + } + } + + @Override + public void remove(FileAccessPartition partition) { + String query = String.format("DROP TABLE %s;", partition.getName()); + int result = jdbcTemplate.update(query); + if (result == 1) { + log.info("Dropped file access partition {}", partition); + } + } +} diff --git a/smart-metastore/src/main/java/org/smartdata/metastore/dao/sqlite/SqliteDaoProvider.java b/smart-metastore/src/main/java/org/smartdata/metastore/dao/sqlite/SqliteDaoProvider.java deleted file mode 100644 index da67ca6ab8b..00000000000 --- a/smart-metastore/src/main/java/org/smartdata/metastore/dao/sqlite/SqliteDaoProvider.java +++ /dev/null @@ -1,44 +0,0 @@ -/** - * 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.smartdata.metastore.dao.sqlite; - -import org.smartdata.metastore.DBPool; -import org.smartdata.metastore.dao.FileInfoDao; -import org.smartdata.metastore.dao.FileStateDao; -import org.smartdata.metastore.dao.SmallFileDao; -import org.smartdata.metastore.dao.impl.DefaultDaoProvider; -import org.springframework.transaction.PlatformTransactionManager; - -public class SqliteDaoProvider extends DefaultDaoProvider { - public SqliteDaoProvider(DBPool dbPool, PlatformTransactionManager transactionManager) { - super(dbPool, transactionManager); - } - - @Override - public FileInfoDao fileInfoDao() { - return new SqliteFileInfoDao(dataSource); - } - - public FileStateDao fileStateDao() { - return new SqliteFileStateDao(dataSource); - } - - public SmallFileDao smallFileDao() { - return new SqliteSmallFileDao(dataSource); - } -} diff --git a/smart-metastore/src/main/java/org/smartdata/metastore/dao/sqlite/SqliteFileStateDao.java b/smart-metastore/src/main/java/org/smartdata/metastore/dao/sqlite/SqliteFileStateDao.java deleted file mode 100644 index 0551dd980ef..00000000000 --- a/smart-metastore/src/main/java/org/smartdata/metastore/dao/sqlite/SqliteFileStateDao.java +++ /dev/null @@ -1,34 +0,0 @@ -/** - * 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.smartdata.metastore.dao.sqlite; - -import org.smartdata.metastore.dao.impl.DefaultFileStateDao; - -import javax.sql.DataSource; - -public class SqliteFileStateDao extends DefaultFileStateDao { - - public SqliteFileStateDao(DataSource dataSource) { - super(dataSource); - } - - protected void renameDirectoryFiles(String oldPath, String newPath) { - String sql = "UPDATE " + tableName + " SET path = ? || SUBSTR(path, ?) WHERE path LIKE ?"; - jdbcTemplate.update(sql, newPath, oldPath.length() + 1, oldPath + "/%"); - } -} diff --git a/smart-metastore/src/main/java/org/smartdata/metastore/dao/sqlite/SqliteSmallFileDao.java b/smart-metastore/src/main/java/org/smartdata/metastore/dao/sqlite/SqliteSmallFileDao.java deleted file mode 100644 index 28729bd69f9..00000000000 --- a/smart-metastore/src/main/java/org/smartdata/metastore/dao/sqlite/SqliteSmallFileDao.java +++ /dev/null @@ -1,34 +0,0 @@ -/** - * 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.smartdata.metastore.dao.sqlite; - -import org.smartdata.metastore.dao.impl.DefaultSmallFileDao; - -import javax.sql.DataSource; - -public class SqliteSmallFileDao extends DefaultSmallFileDao { - - public SqliteSmallFileDao(DataSource dataSource) { - super(dataSource); - } - - protected void renameDirectoryFiles(String oldPath, String newPath) { - String sql = "UPDATE " + tableName + " SET path = ? || SUBSTR(path, ?) WHERE path LIKE ?"; - jdbcTemplate.update(sql, newPath, oldPath.length() + 1, oldPath + "/%"); - } -} diff --git a/smart-metastore/src/main/java/org/smartdata/metastore/db/DBHandlersFactory.java b/smart-metastore/src/main/java/org/smartdata/metastore/db/DBHandlersFactory.java index 8c75436f92f..d9c1cc98ba8 100644 --- a/smart-metastore/src/main/java/org/smartdata/metastore/db/DBHandlersFactory.java +++ b/smart-metastore/src/main/java/org/smartdata/metastore/db/DBHandlersFactory.java @@ -22,46 +22,29 @@ import org.smartdata.metastore.DBType; import org.smartdata.metastore.MetaStoreException; import org.smartdata.metastore.db.metadata.DbMetadataProvider; -import org.smartdata.metastore.db.metadata.MySqlDbMetadataProvider; import org.smartdata.metastore.db.metadata.PostgresDbMetadataProvider; -import org.smartdata.metastore.db.metadata.SqliteDbMetadataProvider; import javax.sql.DataSource; -import static org.smartdata.conf.SmartConfKeys.SMART_METASTORE_LEGACY_MYSQL_SUPPORT_DEFAULT; -import static org.smartdata.conf.SmartConfKeys.SMART_METASTORE_LEGACY_MYSQL_SUPPORT_KEY; import static org.smartdata.conf.SmartConfKeys.SMART_METASTORE_MIGRATION_CHANGELOG_PATH_DEFAULT; import static org.smartdata.conf.SmartConfKeys.SMART_METASTORE_MIGRATION_CHANGELOG_PATH_KEY; public class DBHandlersFactory { - private static final String OLD_MYSQL_LABEL = "old_mysql"; public DbSchemaManager createDbManager(DBPool dbPool, Configuration conf) { String changelogPath = conf.get( SMART_METASTORE_MIGRATION_CHANGELOG_PATH_KEY, SMART_METASTORE_MIGRATION_CHANGELOG_PATH_DEFAULT); - - boolean legacyMysqlSupportEnabled = conf.getBoolean( - SMART_METASTORE_LEGACY_MYSQL_SUPPORT_KEY, - SMART_METASTORE_LEGACY_MYSQL_SUPPORT_DEFAULT); - - String labelsFilter = legacyMysqlSupportEnabled - ? OLD_MYSQL_LABEL - : "!" + OLD_MYSQL_LABEL; - - return new LiquibaseDbSchemaManager(dbPool, changelogPath, labelsFilter); + return new LiquibaseDbSchemaManager(dbPool, changelogPath); } public DbMetadataProvider createDbMetadataProvider(DBPool dbPool, DBType dbType) throws MetaStoreException { DataSource dataSource = dbPool.getDataSource(); switch (dbType) { - case MYSQL: - return new MySqlDbMetadataProvider(dataSource); case POSTGRES: - return new PostgresDbMetadataProvider(dataSource); default: - return new SqliteDbMetadataProvider(dataSource); + return new PostgresDbMetadataProvider(dataSource); } } } diff --git a/smart-metastore/src/main/java/org/smartdata/metastore/db/LiquibaseDbSchemaManager.java b/smart-metastore/src/main/java/org/smartdata/metastore/db/LiquibaseDbSchemaManager.java index 58fb169f340..06a3d6d2072 100644 --- a/smart-metastore/src/main/java/org/smartdata/metastore/db/LiquibaseDbSchemaManager.java +++ b/smart-metastore/src/main/java/org/smartdata/metastore/db/LiquibaseDbSchemaManager.java @@ -35,14 +35,12 @@ public class LiquibaseDbSchemaManager implements DbSchemaManager { private final DBPool pool; private final String changelogPath; - private final String labelFilterExpr; private final Map scopeConfig; - public LiquibaseDbSchemaManager(DBPool pool, String changelogPath, String labelFilterExpr) { + public LiquibaseDbSchemaManager(DBPool pool, String changelogPath) { this.pool = pool; this.changelogPath = changelogPath; - this.labelFilterExpr = labelFilterExpr; this.scopeConfig = new HashMap<>(); scopeConfig.put(Scope.Attr.ui.name(), new LoggerUIService()); @@ -57,8 +55,7 @@ public void initializeDatabase() throws Exception { CommandScope updateCommand = new CommandScope(UpdateCommandStep.COMMAND_NAME) .addArgumentValue(DbUrlConnectionCommandStep.DATABASE_ARG, db) - .addArgumentValue(UpdateCommandStep.CHANGELOG_FILE_ARG, changelogPath) - .addArgumentValue(UpdateCommandStep.LABEL_FILTER_ARG, labelFilterExpr); + .addArgumentValue(UpdateCommandStep.CHANGELOG_FILE_ARG, changelogPath); executeWithLogging(updateCommand); } diff --git a/smart-metastore/src/main/java/org/smartdata/metastore/model/AggregatedAccessCounts.java b/smart-metastore/src/main/java/org/smartdata/metastore/model/AggregatedAccessCounts.java index 9d6afae040a..c3c4ab4d0bb 100644 --- a/smart-metastore/src/main/java/org/smartdata/metastore/model/AggregatedAccessCounts.java +++ b/smart-metastore/src/main/java/org/smartdata/metastore/model/AggregatedAccessCounts.java @@ -27,7 +27,7 @@ @RequiredArgsConstructor public class AggregatedAccessCounts { private long fileId; - private final int accessCount; + private final long accessCount; private final long lastAccessedTimestamp; public AggregatedAccessCounts merge(AggregatedAccessCounts other) { diff --git a/smart-metastore/src/main/java/org/smartdata/metastore/dao/sqlite/SqliteFileInfoDao.java b/smart-metastore/src/main/java/org/smartdata/metastore/model/FileAccessPartition.java similarity index 60% rename from smart-metastore/src/main/java/org/smartdata/metastore/dao/sqlite/SqliteFileInfoDao.java rename to smart-metastore/src/main/java/org/smartdata/metastore/model/FileAccessPartition.java index f3ec1508edb..5318f2214a4 100644 --- a/smart-metastore/src/main/java/org/smartdata/metastore/dao/sqlite/SqliteFileInfoDao.java +++ b/smart-metastore/src/main/java/org/smartdata/metastore/model/FileAccessPartition.java @@ -15,20 +15,19 @@ * See the License for the specific language governing permissions and * limitations under the License. */ -package org.smartdata.metastore.dao.sqlite; +package org.smartdata.metastore.model; -import org.smartdata.metastore.dao.impl.DefaultFileInfoDao; +import lombok.EqualsAndHashCode; +import lombok.Getter; +import lombok.RequiredArgsConstructor; -import javax.sql.DataSource; +import java.time.LocalDate; -public class SqliteFileInfoDao extends DefaultFileInfoDao { - - public SqliteFileInfoDao(DataSource dataSource) { - super(dataSource); - } - - protected void renameDirectoryFiles(String oldPath, String newPath) { - String sql = "UPDATE " + tableName + " SET path = ? || SUBSTR(path, ?) WHERE path LIKE ?"; - jdbcTemplate.update(sql, newPath, oldPath.length() + 1, oldPath + "/%"); - } +@EqualsAndHashCode +@Getter +@RequiredArgsConstructor +public class FileAccessPartition { + private final long id; + private final String name; + private final LocalDate date; } diff --git a/smart-metastore/src/main/java/org/smartdata/metastore/partition/FileAccessPartitionManager.java b/smart-metastore/src/main/java/org/smartdata/metastore/partition/FileAccessPartitionManager.java new file mode 100644 index 00000000000..9c2cf1c891c --- /dev/null +++ b/smart-metastore/src/main/java/org/smartdata/metastore/partition/FileAccessPartitionManager.java @@ -0,0 +1,23 @@ +/** + * 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.smartdata.metastore.partition; + +public interface FileAccessPartitionManager { + + void createNewPartitions(); +} diff --git a/smart-metastore/src/main/java/org/smartdata/metastore/dao/accesscount/AccessCountTableEvictor.java b/smart-metastore/src/main/java/org/smartdata/metastore/partition/FileAccessPartitionManagerImpl.java similarity index 51% rename from smart-metastore/src/main/java/org/smartdata/metastore/dao/accesscount/AccessCountTableEvictor.java rename to smart-metastore/src/main/java/org/smartdata/metastore/partition/FileAccessPartitionManagerImpl.java index d791dab3287..69307613741 100644 --- a/smart-metastore/src/main/java/org/smartdata/metastore/dao/accesscount/AccessCountTableEvictor.java +++ b/smart-metastore/src/main/java/org/smartdata/metastore/partition/FileAccessPartitionManagerImpl.java @@ -15,29 +15,34 @@ * See the License for the specific language governing permissions and * limitations under the License. */ -package org.smartdata.metastore.dao.accesscount; +package org.smartdata.metastore.partition; -import org.slf4j.Logger; -import org.slf4j.LoggerFactory; +import lombok.extern.slf4j.Slf4j; +import org.smartdata.metastore.MetaStore; import org.smartdata.metastore.MetaStoreException; -import org.smartdata.metastore.model.AccessCountTable; +import org.smartdata.metastore.dao.FileAccessPartitionDao; -public abstract class AccessCountTableEvictor { - public static final Logger LOG = LoggerFactory.getLogger(AccessCountTableEvictor.class); - private final AccessCountTableHandler tableDeleter; +import java.time.LocalDateTime; - public AccessCountTableEvictor(AccessCountTableHandler tableDeleter) { - this.tableDeleter = tableDeleter; +@Slf4j +public class FileAccessPartitionManagerImpl implements FileAccessPartitionManager { + + private final FileAccessPartitionDao fileAccessPartitionDao; + + public FileAccessPartitionManagerImpl(MetaStore metaStore) { + this.fileAccessPartitionDao = metaStore.fileAccessPartitionDao(); } - public void dropTable(AccessCountTable accessCountTable) { + @Override + public void createNewPartitions() { try { - tableDeleter.dropTable(accessCountTable); - LOG.debug("Dropped access count table " + accessCountTable.getTableName()); + //create partition for current and next months if they don't exist + LocalDateTime currentDate = LocalDateTime.now(); + fileAccessPartitionDao.create(currentDate); + fileAccessPartitionDao.create(currentDate.plusMonths(1)); } catch (MetaStoreException e) { - LOG.error("Drop access count table {} failed", accessCountTable.getTableName(), e); + log.error("Failed to create partitions", e); + throw new RuntimeException(e); } } - - abstract void evictTables(AccessCountTableDeque tables, long lastAggregatedIntervalEndTimestamp); } diff --git a/smart-metastore/src/main/java/org/smartdata/metastore/partition/FileAccessPartitionService.java b/smart-metastore/src/main/java/org/smartdata/metastore/partition/FileAccessPartitionService.java new file mode 100644 index 00000000000..27794ab93fe --- /dev/null +++ b/smart-metastore/src/main/java/org/smartdata/metastore/partition/FileAccessPartitionService.java @@ -0,0 +1,90 @@ +/** + * 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.smartdata.metastore.partition; + +import lombok.RequiredArgsConstructor; +import lombok.extern.slf4j.Slf4j; +import org.smartdata.metastore.partition.cleanup.FileAccessPartitionRetentionPolicyExecutor; + +import java.util.concurrent.ScheduledExecutorService; +import java.util.concurrent.ScheduledFuture; +import java.util.concurrent.TimeUnit; + +@RequiredArgsConstructor +@Slf4j +public class FileAccessPartitionService { + private static final int CREATE_PARTITION_INTERVAL_DAYS = 30; + private static final int CLEANUP_PARTITION_INTERVAL_DAYS = 30; + private final ScheduledExecutorService scheduledExecutorService; + private final CreatePartitionTask createPartitionTask; + private final CleanupPartitionTask cleanupPartitionTask; + private ScheduledFuture createPartitionFuture; + private ScheduledFuture removePartitionFuture; + + public FileAccessPartitionService(ScheduledExecutorService service, + FileAccessPartitionManager fileAccessPartitionManager, + FileAccessPartitionRetentionPolicyExecutor retentionPolicyExecutor) { + this.scheduledExecutorService = service; + this.createPartitionTask = new CreatePartitionTask(fileAccessPartitionManager); + this.cleanupPartitionTask = new CleanupPartitionTask(retentionPolicyExecutor); + } + + public void start() { + this.createPartitionFuture = scheduledExecutorService.scheduleAtFixedRate( + createPartitionTask, 0, CREATE_PARTITION_INTERVAL_DAYS, TimeUnit.DAYS); + this.removePartitionFuture = scheduledExecutorService.scheduleAtFixedRate( + cleanupPartitionTask, 0, CLEANUP_PARTITION_INTERVAL_DAYS, TimeUnit.DAYS); + } + + public void stop() { + if (createPartitionFuture != null) { + createPartitionFuture.cancel(true); + } + if (removePartitionFuture != null) { + removePartitionFuture.cancel(true); + } + } + + @RequiredArgsConstructor + private static class CreatePartitionTask implements Runnable { + private final FileAccessPartitionManager partitionManager; + + @Override + public void run() { + try { + partitionManager.createNewPartitions(); + } catch (Exception e) { + log.error("CreatePartitionTask failed", e); + } + } + } + + @RequiredArgsConstructor + private static class CleanupPartitionTask implements Runnable { + private final FileAccessPartitionRetentionPolicyExecutor retentionPolicyExecutor; + + @Override + public void run() { + try { + retentionPolicyExecutor.cleanup(); + } catch (Exception e) { + log.error("CleanupPartitionTask failed", e); + } + } + } +} diff --git a/smart-metastore/src/main/java/org/smartdata/metastore/partition/cleanup/FileAccessPartitionRetentionPolicyExecutor.java b/smart-metastore/src/main/java/org/smartdata/metastore/partition/cleanup/FileAccessPartitionRetentionPolicyExecutor.java new file mode 100644 index 00000000000..0899cb18783 --- /dev/null +++ b/smart-metastore/src/main/java/org/smartdata/metastore/partition/cleanup/FileAccessPartitionRetentionPolicyExecutor.java @@ -0,0 +1,25 @@ +/** + * 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.smartdata.metastore.partition.cleanup; + +public interface FileAccessPartitionRetentionPolicyExecutor { + + void cleanup(); + + FileAccessPartitionRetentionPolicyType getPolicyType(); +} diff --git a/smart-metastore/src/main/java/org/smartdata/metastore/partition/cleanup/FileAccessPartitionRetentionPolicyExecutorFactory.java b/smart-metastore/src/main/java/org/smartdata/metastore/partition/cleanup/FileAccessPartitionRetentionPolicyExecutorFactory.java new file mode 100644 index 00000000000..9032b066fdd --- /dev/null +++ b/smart-metastore/src/main/java/org/smartdata/metastore/partition/cleanup/FileAccessPartitionRetentionPolicyExecutorFactory.java @@ -0,0 +1,66 @@ +/** + * 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.smartdata.metastore.partition.cleanup; + +import lombok.RequiredArgsConstructor; +import org.smartdata.conf.SmartConf; +import org.smartdata.metastore.MetaStore; +import org.smartdata.metastore.partition.cleanup.impl.MonthCountFileAccessPartitionRetentionPolicyExecutor; + +import static org.smartdata.conf.SmartConfKeys.SMART_FILE_ACCESS_PARTITIONS_RETENTION_COUNT_DEFAULT; +import static org.smartdata.conf.SmartConfKeys.SMART_FILE_ACCESS_PARTITIONS_RETENTION_COUNT_KEY; +import static org.smartdata.conf.SmartConfKeys.SMART_FILE_ACCESS_PARTITIONS_RETENTION_POLICY_KEY; + +@RequiredArgsConstructor +public class FileAccessPartitionRetentionPolicyExecutorFactory { + + private final MetaStore metaStore; + + public FileAccessPartitionRetentionPolicyExecutor createPolicyExecutor(SmartConf conf) { + FileAccessPartitionRetentionPolicyType policyType = + conf.getEnum(SMART_FILE_ACCESS_PARTITIONS_RETENTION_POLICY_KEY, + FileAccessPartitionRetentionPolicyType.MONTH_COUNT); + int retentionCount = conf.getInt(SMART_FILE_ACCESS_PARTITIONS_RETENTION_COUNT_KEY, + SMART_FILE_ACCESS_PARTITIONS_RETENTION_COUNT_DEFAULT); + switch (policyType) { + case MONTH_COUNT: + default: + return new MonthCountFileAccessPartitionRetentionPolicyExecutor( + metaStore.fileAccessPartitionDao(), + retentionCount); + } + } +} + + + + + + + + + + + + + + + + + + diff --git a/smart-metastore/src/main/java/org/smartdata/metastore/partition/cleanup/FileAccessPartitionRetentionPolicyType.java b/smart-metastore/src/main/java/org/smartdata/metastore/partition/cleanup/FileAccessPartitionRetentionPolicyType.java new file mode 100644 index 00000000000..8282c490214 --- /dev/null +++ b/smart-metastore/src/main/java/org/smartdata/metastore/partition/cleanup/FileAccessPartitionRetentionPolicyType.java @@ -0,0 +1,22 @@ +/** + * 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.smartdata.metastore.partition.cleanup; + +public enum FileAccessPartitionRetentionPolicyType { + MONTH_COUNT +} diff --git a/smart-metastore/src/main/java/org/smartdata/metastore/partition/cleanup/impl/MonthCountFileAccessPartitionRetentionPolicyExecutor.java b/smart-metastore/src/main/java/org/smartdata/metastore/partition/cleanup/impl/MonthCountFileAccessPartitionRetentionPolicyExecutor.java new file mode 100644 index 00000000000..f976d738d26 --- /dev/null +++ b/smart-metastore/src/main/java/org/smartdata/metastore/partition/cleanup/impl/MonthCountFileAccessPartitionRetentionPolicyExecutor.java @@ -0,0 +1,61 @@ +/** + * 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.smartdata.metastore.partition.cleanup.impl; + +import lombok.RequiredArgsConstructor; +import lombok.extern.slf4j.Slf4j; +import org.smartdata.metastore.dao.FileAccessPartitionDao; +import org.smartdata.metastore.model.FileAccessPartition; +import org.smartdata.metastore.partition.cleanup.FileAccessPartitionRetentionPolicyExecutor; +import org.smartdata.metastore.partition.cleanup.FileAccessPartitionRetentionPolicyType; + +import java.util.List; + +@Slf4j +@RequiredArgsConstructor +public class MonthCountFileAccessPartitionRetentionPolicyExecutor + implements FileAccessPartitionRetentionPolicyExecutor { + + private final FileAccessPartitionDao fileAccessPartitionDao; + private final int monthCount; + + @Override + public void cleanup() { + if (monthCount <= 0) { + return; + } + List fileAccessPartitions = fileAccessPartitionDao.getAll(); + //we should remove + 1 partition, because we also have partition for the next month + if (fileAccessPartitions.size() > monthCount + 1) { + List partitionsToRemove = + fileAccessPartitions.subList(monthCount, fileAccessPartitions.size()); + for (FileAccessPartition fileAccessPartition : partitionsToRemove) { + fileAccessPartitionDao.remove(fileAccessPartition); + } + } + log.info( + "File access partitions were cleanup successfully by retention policy: {}," + + " retention count: {}", + getPolicyType(), monthCount); + } + + @Override + public FileAccessPartitionRetentionPolicyType getPolicyType() { + return FileAccessPartitionRetentionPolicyType.MONTH_COUNT; + } +} diff --git a/smart-metastore/src/main/java/org/smartdata/metastore/utils/MetaStoreUtils.java b/smart-metastore/src/main/java/org/smartdata/metastore/utils/MetaStoreUtils.java index 11f0e25ed5f..f218f5ae471 100644 --- a/smart-metastore/src/main/java/org/smartdata/metastore/utils/MetaStoreUtils.java +++ b/smart-metastore/src/main/java/org/smartdata/metastore/utils/MetaStoreUtils.java @@ -54,11 +54,10 @@ public class MetaStoreUtils { static final Logger LOG = LoggerFactory.getLogger(MetaStoreUtils.class); public static final List SSM_TABLES = Lists.newArrayList( - "access_count_table", - "blank_access_count_info", "cached_file", "ec_policy", "file", + "file_access", "storage", "storage_hist", "storage_policy", @@ -134,13 +133,6 @@ private static Properties loadDruidConfig(SmartConf conf, File cpConfigFile) p.setProperty("url", url); } - String purl = p.getProperty("url"); - if (purl == null || purl.isEmpty()) { - purl = getDefaultSqliteDB(); // For testing - p.setProperty("url", purl); - LOG.warn("Database URL not specified, using " + purl); - } - try { conf.getPasswordFromHadoop(SmartConfKeys.SMART_METASTORE_PASSWORD) .filter(password -> !StringUtils.isBlank(password)) @@ -185,24 +177,11 @@ private static Properties loadDefaultDruidConfig(SmartConf conf, File cpConfigFi return properties; } - /** - * This default behavior provided here is mainly for convenience. - */ - private static String getDefaultSqliteDB() { - String absFilePath = System.getProperty("user.home") - + "/smart-test-default.db"; - return MetaStoreUtils.SQLITE_URL_PREFIX + absFilePath; - } - private static DBType getDbType(DBPool dbPool) throws MetaStoreException { try (Connection connection = dbPool.getConnection()) { String driver = connection.getMetaData().getDriverName(); driver = driver.toLowerCase(); - if (driver.contains("sqlite")) { - return DBType.SQLITE; - } else if (driver.contains("mysql")) { - return DBType.MYSQL; - } else if (driver.contains("postgres")) { + if (driver.contains("postgres")) { return DBType.POSTGRES; } else { throw new MetaStoreException("Unknown database: " + driver); diff --git a/smart-metastore/src/main/resources/db/changelog/changelog-1.init-db.xml b/smart-metastore/src/main/resources/db/changelog/changelog-1.init-db.xml index 0b0336904cf..23696dcc968 100644 --- a/smart-metastore/src/main/resources/db/changelog/changelog-1.init-db.xml +++ b/smart-metastore/src/main/resources/db/changelog/changelog-1.init-db.xml @@ -8,13 +8,6 @@ http://www.liquibase.org/xml/ns/dbchangelog/dbchangelog-latest.xsd http://www.liquibase.org/xml/ns/dbchangelog-ext http://www.liquibase.org/xml/ns/dbchangelog/dbchangelog-ext.xsd http://www.liquibase.org/xml/ns/pro http://www.liquibase.org/xml/ns/pro/liquibase-pro-latest.xsd"> - - - SET GLOBAL innodb_file_format=barracuda; - SET GLOBAL innodb_file_per_table=true; - SET GLOBAL innodb_large_prefix = ON; - - @@ -76,15 +69,6 @@ - - - - - - ALTER TABLE cached_file ROW_FORMAT= DYNAMIC ENGINE=INNODB; - - - @@ -170,9 +154,6 @@ - - - @@ -499,15 +480,6 @@ - - - - - - ALTER TABLE file_diff ROW_FORMAT= DYNAMIC ENGINE=INNODB; - - - @@ -517,9 +489,6 @@ - - - @@ -734,9 +703,6 @@ - - - diff --git a/smart-metastore/src/main/resources/db/changelog/changelog-3.create-user-activities-table.xml b/smart-metastore/src/main/resources/db/changelog/changelog-3.create-user-activities-table.xml index d39cdff6f2e..fc3e17355fd 100644 --- a/smart-metastore/src/main/resources/db/changelog/changelog-3.create-user-activities-table.xml +++ b/smart-metastore/src/main/resources/db/changelog/changelog-3.create-user-activities-table.xml @@ -40,13 +40,6 @@ - - - - \ No newline at end of file diff --git a/smart-metastore/src/main/resources/db/changelog/changelog-5.add-partitioning.xml b/smart-metastore/src/main/resources/db/changelog/changelog-5.add-partitioning.xml new file mode 100644 index 00000000000..b7cc1bda24b --- /dev/null +++ b/smart-metastore/src/main/resources/db/changelog/changelog-5.add-partitioning.xml @@ -0,0 +1,55 @@ + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + select count(*) from pg_proc where proname = 'create_file_access_partition' + + + + select create_file_access_partition(cast(now() as date)); + + + + + + + + + + + + + + \ No newline at end of file diff --git a/smart-metastore/src/main/resources/db/changelog/changelog-root.xml b/smart-metastore/src/main/resources/db/changelog/changelog-root.xml index 13aa64639bc..55021b0e158 100644 --- a/smart-metastore/src/main/resources/db/changelog/changelog-root.xml +++ b/smart-metastore/src/main/resources/db/changelog/changelog-root.xml @@ -12,4 +12,5 @@ + \ No newline at end of file diff --git a/smart-metastore/src/main/resources/db/changelog/sql/create_file_access_partition.sql b/smart-metastore/src/main/resources/db/changelog/sql/create_file_access_partition.sql new file mode 100644 index 00000000000..28bfbd23790 --- /dev/null +++ b/smart-metastore/src/main/resources/db/changelog/sql/create_file_access_partition.sql @@ -0,0 +1,30 @@ +CREATE OR REPLACE FUNCTION create_file_access_partition(input_date timestamp) + RETURNS INTEGER AS ' + DECLARE + current_date_part DATE; + current_date_part_text TEXT; + partition_table_name TEXT; + first_day_of_month DATE; + last_day_of_month DATE; + result INTEGER; + create_query TEXT; + BEGIN + result := 0; + current_date_part := CAST(DATE_TRUNC(''month'', input_date::date) AS DATE); + current_date_part_text := REGEXP_REPLACE(current_date_part::TEXT, ''-'',''_'',''g''); + partition_table_name := FORMAT(''file_access_%s'', current_date_part_text::TEXT); + IF (TO_REGCLASS(partition_table_name::TEXT) ISNULL) THEN + first_day_of_month := current_date_part; + last_day_of_month := current_date_part + ''1 month''::INTERVAL; + result := 1; + RAISE NOTICE ''table: %'', partition_table_name; + create_query := FORMAT( + ''CREATE TABLE %I PARTITION OF file_access FOR VALUES FROM (extract(epoch from %L::DATE) * 1000) TO (extract(epoch from %L::DATE) * 1000);'', + partition_table_name, first_day_of_month, last_day_of_month); + RAISE NOTICE ''query: %'', create_query; + EXECUTE create_query; + EXECUTE FORMAT(''CREATE INDEX %1$s__access_time ON %1$I (access_time);'', partition_table_name); + END IF; + RETURN result; + END; +' LANGUAGE plpgsql; diff --git a/smart-metastore/src/test/java/org/smartdata/metastore/SqliteTestDaoBase.java b/smart-metastore/src/test/java/org/smartdata/metastore/SqliteTestDaoBase.java deleted file mode 100644 index 8e932e4dbfa..00000000000 --- a/smart-metastore/src/test/java/org/smartdata/metastore/SqliteTestDaoBase.java +++ /dev/null @@ -1,36 +0,0 @@ -/** - * 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.smartdata.metastore; - -import org.junit.runners.Parameterized.Parameters; -import org.sqlite.JDBC; - -import static org.smartdata.metastore.TestDBUtil.getUniqueSqliteUrl; - -/** - * Limits the number of databases for testing to sqlite for tests - * that don't require a complete set of DBMS. - */ -public abstract class SqliteTestDaoBase extends TestDaoBase { - @Parameters(name = "{0}") - public static Object[] parameters() { - return new Object[][]{ - {DBType.SQLITE, JDBC.class.getName(), getUniqueSqliteUrl()} - }; - } -} diff --git a/smart-metastore/src/test/java/org/smartdata/metastore/TestDBUtil.java b/smart-metastore/src/test/java/org/smartdata/metastore/TestDBUtil.java index d7f7c10a65a..2b6f906340e 100644 --- a/smart-metastore/src/test/java/org/smartdata/metastore/TestDBUtil.java +++ b/smart-metastore/src/test/java/org/smartdata/metastore/TestDBUtil.java @@ -19,10 +19,8 @@ import lombok.extern.slf4j.Slf4j; import org.apache.hadoop.conf.Configuration; -import org.smartdata.metastore.dao.accesscount.AccessCountTableDeque; import org.smartdata.metastore.db.DBHandlersFactory; import org.smartdata.metastore.db.DbSchemaManager; -import org.smartdata.metastore.model.AccessCountTable; import javax.sql.DataSource; @@ -37,7 +35,6 @@ import java.sql.DriverManager; import java.sql.SQLException; import java.util.UUID; -import java.util.concurrent.TimeUnit; import static org.smartdata.metastore.utils.MetaStoreUtils.SQLITE_URL_PREFIX; @@ -60,20 +57,6 @@ public static String getUniqueDBFilePath() { return getUniqueFilePath() + ".db"; } - public static String getUniqueSqliteUrl() { - String dbFile = getUniqueDBFilePath(); - new File(dbFile).deleteOnExit(); - String url = SQLITE_URL_PREFIX + getUniqueDBFilePath(); - log.info("sqlite url: {}", url); - return url; - } - - public static void addAccessCountTableToDeque( - AccessCountTableDeque deque, AccessCountTable table) throws Exception { - deque.addAndNotifyListener(table) - .get(1, TimeUnit.SECONDS); - } - /** * Get an initialized empty Sqlite database file path. * diff --git a/smart-metastore/src/test/java/org/smartdata/metastore/TestDaoBase.java b/smart-metastore/src/test/java/org/smartdata/metastore/TestDaoBase.java index f9363ab84c8..f0b39d24702 100644 --- a/smart-metastore/src/test/java/org/smartdata/metastore/TestDaoBase.java +++ b/smart-metastore/src/test/java/org/smartdata/metastore/TestDaoBase.java @@ -33,7 +33,6 @@ import org.smartdata.metastore.db.metadata.DbMetadataProvider; import org.springframework.jdbc.support.JdbcTransactionManager; import org.springframework.transaction.PlatformTransactionManager; -import org.sqlite.JDBC; import org.testcontainers.jdbc.ContainerDatabaseDriver; import java.io.InputStream; @@ -41,7 +40,6 @@ import static com.alibaba.druid.pool.DruidDataSourceFactory.PROP_DRIVERCLASSNAME; import static com.alibaba.druid.pool.DruidDataSourceFactory.PROP_URL; -import static org.smartdata.metastore.TestDBUtil.getUniqueSqliteUrl; @RunWith(Parameterized.class) public abstract class TestDaoBase { @@ -62,12 +60,9 @@ public abstract class TestDaoBase { @Parameters(name = "{0}") public static Object[] parameters() { - return new Object[][] { - {DBType.SQLITE, JDBC.class.getName(), getUniqueSqliteUrl()}, - {DBType.MYSQL, ContainerDatabaseDriver.class.getName(), - "jdbc:tc:mysql:5.7.34:///ssm_mysql"}, + return new Object[][]{ {DBType.POSTGRES, ContainerDatabaseDriver.class.getName(), - "jdbc:tc:postgresql:9.6.8:///ssm_postgres"}, + "jdbc:tc:postgresql:12.19:///ssm_postgres"}, }; } diff --git a/smart-metastore/src/test/java/org/smartdata/metastore/TestDruid.java b/smart-metastore/src/test/java/org/smartdata/metastore/TestDruid.java index 10357ab1ccb..5696f794f33 100644 --- a/smart-metastore/src/test/java/org/smartdata/metastore/TestDruid.java +++ b/smart-metastore/src/test/java/org/smartdata/metastore/TestDruid.java @@ -21,20 +21,22 @@ import org.junit.Assert; import org.junit.Test; import org.smartdata.metastore.dao.DaoProvider; -import org.smartdata.metastore.dao.sqlite.SqliteDaoProvider; +import org.smartdata.metastore.dao.postgres.PostgresDaoProvider; import org.smartdata.metastore.db.DBHandlersFactory; import org.smartdata.metastore.db.DbSchemaManager; import org.smartdata.metastore.db.metadata.DbMetadataProvider; -import org.smartdata.metastore.db.metadata.SqliteDbMetadataProvider; -import org.smartdata.metastore.utils.MetaStoreUtils; import org.smartdata.model.RuleInfo; import org.smartdata.model.RuleState; import org.springframework.jdbc.support.JdbcTransactionManager; import org.springframework.transaction.PlatformTransactionManager; +import org.testcontainers.jdbc.ContainerDatabaseDriver; import java.io.InputStream; import java.util.Properties; +import static com.alibaba.druid.pool.DruidDataSourceFactory.PROP_DRIVERCLASSNAME; +import static com.alibaba.druid.pool.DruidDataSourceFactory.PROP_URL; + public class TestDruid { @Test @@ -43,22 +45,20 @@ public void test() throws Exception { .getResourceAsStream("druid-template.xml"); Properties p = new Properties(); p.loadFromXML(in); - - String dbFile = TestDBUtil.getUniqueEmptySqliteDBFile(); - String url = MetaStoreUtils.SQLITE_URL_PREFIX + dbFile; - p.setProperty("url", url); - + p.setProperty(PROP_URL, "jdbc:tc:postgresql:12.19:///ssm_postgres"); + p.setProperty(PROP_DRIVERCLASSNAME, ContainerDatabaseDriver.class.getName()); DruidPool druidPool = new DruidPool(p); - DbSchemaManager dbSchemaManager = new DBHandlersFactory() + DBHandlersFactory dbHandlersFactory = new DBHandlersFactory(); + DbSchemaManager dbSchemaManager = dbHandlersFactory .createDbManager(druidPool, new Configuration()); PlatformTransactionManager transactionManager = new JdbcTransactionManager(druidPool.getDataSource()); - DaoProvider daoProvider = new SqliteDaoProvider(druidPool, transactionManager); - DbMetadataProvider dbMetadataProvider = new SqliteDbMetadataProvider( - druidPool.getDataSource()); + DaoProvider daoProvider = new PostgresDaoProvider(druidPool, transactionManager); + DbMetadataProvider dbMetadataProvider = + dbHandlersFactory.createDbMetadataProvider(druidPool, DBType.POSTGRES); MetaStore adapter = new MetaStore( druidPool, dbSchemaManager, daoProvider, dbMetadataProvider, transactionManager); - + dbSchemaManager.initializeDatabase(); String rule = "file : accessCount(10m) > 20 \n\n" + "and length() > 3 | cache"; long submitTime = System.currentTimeMillis(); diff --git a/smart-metastore/src/test/java/org/smartdata/metastore/TestRulesTable.java b/smart-metastore/src/test/java/org/smartdata/metastore/TestRulesTable.java index ebce1b0ad4f..223b6378321 100644 --- a/smart-metastore/src/test/java/org/smartdata/metastore/TestRulesTable.java +++ b/smart-metastore/src/test/java/org/smartdata/metastore/TestRulesTable.java @@ -27,7 +27,7 @@ /** * Tests for table 'rules'. */ -public class TestRulesTable extends SqliteTestDaoBase { +public class TestRulesTable extends TestDaoBase { /** * Insert rules into table and retrieve them back. diff --git a/smart-metastore/src/test/java/org/smartdata/metastore/TestSqliteDB.java b/smart-metastore/src/test/java/org/smartdata/metastore/TestSqliteDB.java deleted file mode 100644 index f68d0994ed3..00000000000 --- a/smart-metastore/src/test/java/org/smartdata/metastore/TestSqliteDB.java +++ /dev/null @@ -1,119 +0,0 @@ -/** - * 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.smartdata.metastore; - -import org.junit.Assert; -import org.junit.Test; -import org.smartdata.metastore.utils.MetaStoreUtils; - -import java.sql.Connection; -import java.sql.ResultSet; -import java.sql.Statement; -import java.util.ArrayList; -import java.util.List; - -/** Test operations with sqlite database. */ -public class TestSqliteDB extends SqliteTestDaoBase { - @Test - public void testDropTables() throws Exception { - Connection conn = druidPool.getConnection(); - Statement s = conn.createStatement(); - metaStore.dropAllTables(); - for (int i = 0; i < 10; i++) { - metaStore.execute("DROP TABLE IF EXISTS tb_" + i + ";"); - metaStore.execute("CREATE TABLE tb_" + i + " (a INT(11));"); - } - String dbUrl = conn.getMetaData().getURL(); - if (dbUrl.startsWith(MetaStoreUtils.SQLITE_URL_PREFIX)) { - ResultSet rs = s.executeQuery("select tbl_name from sqlite_master;"); - List list = new ArrayList<>(); - while (rs.next()) { - list.add(rs.getString(1)); - } - metaStore.dropAllTables(); - rs = s.executeQuery("select tbl_name from sqlite_master;"); - List list1 = new ArrayList<>(); - while (rs.next()) { - list1.add(rs.getString(1)); - } - Assert.assertEquals(10, list.size() - list1.size()); - } else { - String dbName; - if (dbUrl.contains("?")) { - dbName = dbUrl.substring(dbUrl.indexOf("/", 13) + 1, dbUrl.indexOf("?")); - } else { - dbName = dbUrl.substring(dbUrl.lastIndexOf("/") + 1, dbUrl.length()); - } - ResultSet rs = - s.executeQuery( - "SELECT TABLE_NAME FROM " - + "INFORMATION_SCHEMA.TABLES WHERE TABLE_SCHEMA = '" - + dbName - + "';"); - List list = new ArrayList<>(); - while (rs.next()) { - list.add(rs.getString(1)); - } - metaStore.dropAllTables(); - rs = - s.executeQuery( - "SELECT TABLE_NAME FROM " - + "INFORMATION_SCHEMA.TABLES WHERE TABLE_SCHEMA = '" - + dbName - + "';"); - List list1 = new ArrayList<>(); - while (rs.next()) { - list1.add(rs.getString(1)); - } - Assert.assertEquals(10, list.size() - list1.size()); - } - conn.close(); - } - - @Test - public void testDBBlankStatements() throws Exception { - String[] presqls = - new String[]{ - "INSERT INTO rule (state, rule_text, submit_time, checked_count, " - + "generated_cmdlets) VALUES (0, 'file: every 1s \n" - + " | " - + "accessCount(5s) > 3 | cache', 1494903787619, 0, 0);" - }; - - for (int i = 0; i < presqls.length; i++) { - String sql = presqls[i]; - metaStore.execute(sql); - } - - String[] sqls = - new String[]{ - "DROP TABLE IF EXISTS VIR_ACC_CNT_TAB_1_accessCount_5000;", - "CREATE TABLE VIR_ACC_CNT_TAB_1_accessCount_5000 " - + "AS SELECT * FROM blank_access_count_info;", - "SELECT fid from VIR_ACC_CNT_TAB_1_accessCount_5000;", - "SELECT path FROM file WHERE (fid IN (SELECT fid FROM " - + "VIR_ACC_CNT_TAB_1_accessCount_5000 WHERE ((count > 3))));" - }; - - for (int i = 0; i < sqls.length * 3; i++) { - int idx = i % sqls.length; - String sql = sqls[idx]; - metaStore.execute(sql); - } - } -} diff --git a/smart-metastore/src/test/java/org/smartdata/metastore/accesscount/failover/impl/FailAccessCountFailoverTest.java b/smart-metastore/src/test/java/org/smartdata/metastore/accesscount/failover/impl/FailAccessCountFailoverTest.java new file mode 100644 index 00000000000..cee0428da70 --- /dev/null +++ b/smart-metastore/src/test/java/org/smartdata/metastore/accesscount/failover/impl/FailAccessCountFailoverTest.java @@ -0,0 +1,47 @@ +/** + * 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.smartdata.metastore.accesscount.failover.impl; + +import org.junit.Test; +import org.smartdata.metastore.accesscount.failover.AccessCountContext; +import org.smartdata.metastore.accesscount.failover.Failover; +import org.smartdata.metastore.model.AggregatedAccessCounts; + +import java.util.ArrayList; +import java.util.Collections; +import java.util.List; + +import static org.junit.Assert.assertThrows; + +public class FailAccessCountFailoverTest { + private Failover accessCountFailover; + + @Test + public void testExecuteWithoutExceedingOverMaxRetries() { + long currentTimeMillis = System.currentTimeMillis(); + accessCountFailover = new Failover() {}; + List accessCounts = new ArrayList<>(Collections.singletonList( + new AggregatedAccessCounts(1, 1, currentTimeMillis))); + AccessCountContext context = new AccessCountContext(accessCounts); + assertThrows(RuntimeException.class, () -> { + accessCountFailover.execute(ctx -> { + throw new RuntimeException("error"); + }, context); + }); + } +} diff --git a/smart-metastore/src/test/java/org/smartdata/metastore/accesscount/failover/impl/RetryAccessCountFailoverTest.java b/smart-metastore/src/test/java/org/smartdata/metastore/accesscount/failover/impl/RetryAccessCountFailoverTest.java new file mode 100644 index 00000000000..e9a39ffb548 --- /dev/null +++ b/smart-metastore/src/test/java/org/smartdata/metastore/accesscount/failover/impl/RetryAccessCountFailoverTest.java @@ -0,0 +1,70 @@ +/** + * 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.smartdata.metastore.accesscount.failover.impl; + +import org.junit.Test; +import org.smartdata.metastore.accesscount.failover.AccessCountContext; +import org.smartdata.metastore.accesscount.failover.Failover; +import org.smartdata.metastore.model.AggregatedAccessCounts; + +import java.util.ArrayList; +import java.util.Collections; +import java.util.List; + +import static org.junit.Assert.assertEquals; +import static org.junit.Assert.assertThrows; + +public class RetryAccessCountFailoverTest { + private Failover accessCountFailover; + + @Test + public void testExecuteWithoutExceedingOverMaxRetries() { + int maxRetries = 3; + int attemptsCount = 0; + long currentTimeMillis = System.currentTimeMillis(); + accessCountFailover = new RetryAccessCountFailover(maxRetries); + List accessCounts = new ArrayList<>(Collections.singletonList( + new AggregatedAccessCounts(1, 1, currentTimeMillis))); + AccessCountContext context = new AccessCountContext(accessCounts); + for (int i = 1; i <= maxRetries; i++) { + accessCountFailover.execute(ctx -> { + throw new RuntimeException("error"); + }, context); + attemptsCount = i; + } + assertEquals(maxRetries, attemptsCount); + } + + @Test + public void testExecuteWithExceedingOverMaxRetries() { + RuntimeException error = new RuntimeException("error"); + int maxRetries = 3; + long currentTimeMillis = System.currentTimeMillis(); + accessCountFailover = new RetryAccessCountFailover(maxRetries); + List accessCounts = new ArrayList<>(Collections.singletonList( + new AggregatedAccessCounts(1, 1, currentTimeMillis))); + AccessCountContext context = new AccessCountContext(accessCounts); + assertThrows(RuntimeException.class, () -> { + for (int i = 1; i <= maxRetries + 1; i++) { + accessCountFailover.execute(ctx -> { + throw error; + }, context); + } + }); + } +} diff --git a/smart-metastore/src/test/java/org/smartdata/metastore/dao/TestCacheFileDao.java b/smart-metastore/src/test/java/org/smartdata/metastore/dao/TestCacheFileDao.java index 6615aedafec..1cabcd4e314 100644 --- a/smart-metastore/src/test/java/org/smartdata/metastore/dao/TestCacheFileDao.java +++ b/smart-metastore/src/test/java/org/smartdata/metastore/dao/TestCacheFileDao.java @@ -56,8 +56,8 @@ public void testUpdateCachedFiles() { "testPath2", 2000L, 3000L, 200); cacheFileDao.insert(second); List accessCounts = new ArrayList<>(); - accessCounts.add(new AggregatedAccessCounts(80L, 2, 4000L)); - accessCounts.add(new AggregatedAccessCounts(90L, 2, 5000L)); + accessCounts.add(new AggregatedAccessCounts(80L, 2, 4000L)); + accessCounts.add(new AggregatedAccessCounts(90L, 2, 5000L)); accessCounts.add(new AggregatedAccessCounts(100L, 2, 9000L)); // Sync status diff --git a/smart-metastore/src/test/java/org/smartdata/metastore/dao/accesscount/TestAccessCountTableEvictor.java b/smart-metastore/src/test/java/org/smartdata/metastore/dao/accesscount/TestAccessCountTableEvictor.java deleted file mode 100644 index efdb4f71bf9..00000000000 --- a/smart-metastore/src/test/java/org/smartdata/metastore/dao/accesscount/TestAccessCountTableEvictor.java +++ /dev/null @@ -1,129 +0,0 @@ -/** - * 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.smartdata.metastore.dao.accesscount; - -import org.junit.After; -import org.junit.Assert; -import org.junit.Before; -import org.junit.Test; -import org.smartdata.metastore.model.AccessCountTable; - -import java.util.concurrent.ExecutorService; -import java.util.concurrent.Executors; - -import static org.smartdata.metastore.TestDBUtil.addAccessCountTableToDeque; -import static org.smartdata.metastore.utils.Constants.ONE_SECOND_IN_MILLIS; - -public class TestAccessCountTableEvictor { - private ExecutorService executorService; - private AccessCountTableHandler tableHandler; - - @Before - public void setUp() { - tableHandler = new NoOpAccessCountTableHandler(); - executorService = Executors.newSingleThreadExecutor(); - } - - @After - public void shutdown() { - executorService.shutdown(); - } - - @Test - public void testCountEvictor() { - CountTableEvictor countEvictor = new CountTableEvictor(tableHandler, 2); - AccessCountTableDeque tableDeque = new AccessCountTableDeque(countEvictor); - - tableDeque.add(new AccessCountTable(0L, 1L)); - countEvictor.evictTables(tableDeque, 0L); - Assert.assertEquals(1, tableDeque.size()); - - tableDeque.add(new AccessCountTable(1L, 2L)); - countEvictor.evictTables(tableDeque, 0L); - Assert.assertEquals(2, tableDeque.size()); - - tableDeque.add(new AccessCountTable(3L, 4L)); - countEvictor.evictTables(tableDeque, 0L); - Assert.assertEquals(3, tableDeque.size()); - - AccessCountTable firstExpectedTable = new AccessCountTable(4L, 59 * ONE_SECOND_IN_MILLIS); - tableDeque.add(firstExpectedTable); - countEvictor.evictTables(tableDeque, 0L); - Assert.assertEquals(4, tableDeque.size()); - - AccessCountTable secondExpectedTable = new AccessCountTable( - 59 * ONE_SECOND_IN_MILLIS, 60 * ONE_SECOND_IN_MILLIS); - tableDeque.add(secondExpectedTable); - countEvictor.evictTables(tableDeque, ONE_SECOND_IN_MILLIS); - Assert.assertEquals(2, tableDeque.size()); - - Assert.assertTrue(tableDeque.contains(firstExpectedTable)); - Assert.assertTrue(tableDeque.contains(secondExpectedTable)); - } - - @Test - public void testDontEvictIfNotAggregatedYet() throws Exception { - AccessCountTableDeque secondDeque = buildSecondDeque(1); - - addAccessCountTableToDeque(secondDeque, new AccessCountTable(0L, 1L)); - Assert.assertEquals(1, secondDeque.size()); - - addAccessCountTableToDeque(secondDeque, new AccessCountTable(1L, 2L)); - Assert.assertEquals(2, secondDeque.size()); - - addAccessCountTableToDeque(secondDeque, new AccessCountTable(2L, 3L)); - Assert.assertEquals(3, secondDeque.size()); - - AccessCountTable lastFirstMinuteTable = new AccessCountTable( - 59 * ONE_SECOND_IN_MILLIS, 60 * ONE_SECOND_IN_MILLIS); - addAccessCountTableToDeque(secondDeque, lastFirstMinuteTable); - - Assert.assertEquals(1, secondDeque.size()); - Assert.assertTrue(secondDeque.contains(lastFirstMinuteTable)); - } - - @Test - public void testDontEvictDuringAggregationIfThresholdNotMet() throws Exception { - AccessCountTableDeque secondDeque = buildSecondDeque(10); - - addAccessCountTableToDeque(secondDeque, new AccessCountTable(0L, 1L)); - Assert.assertEquals(1, secondDeque.size()); - - addAccessCountTableToDeque(secondDeque, new AccessCountTable(1L, 2L)); - Assert.assertEquals(2, secondDeque.size()); - - addAccessCountTableToDeque(secondDeque, new AccessCountTable(2L, 3L)); - Assert.assertEquals(3, secondDeque.size()); - - AccessCountTable lastFirstMinuteTable = new AccessCountTable( - 59 * ONE_SECOND_IN_MILLIS, 60 * ONE_SECOND_IN_MILLIS); - addAccessCountTableToDeque(secondDeque, lastFirstMinuteTable); - - Assert.assertEquals(4, secondDeque.size()); - } - - private AccessCountTableDeque buildSecondDeque(int evictThreshold) { - AccessCountTableDeque minuteDeque = new AccessCountTableDeque( - new CountTableEvictor(tableHandler, 999)); - TableAddOpListener minuteTableListener = - TableAddOpListener.perMinute(minuteDeque, tableHandler, executorService); - - AccessCountTableEvictor secondEvictor = new CountTableEvictor(tableHandler, evictThreshold); - return new AccessCountTableDeque(secondEvictor, minuteTableListener); - } -} diff --git a/smart-metastore/src/test/java/org/smartdata/metastore/dao/accesscount/TestAccessCountTableManager.java b/smart-metastore/src/test/java/org/smartdata/metastore/dao/accesscount/TestAccessCountTableManager.java deleted file mode 100644 index 9682291660c..00000000000 --- a/smart-metastore/src/test/java/org/smartdata/metastore/dao/accesscount/TestAccessCountTableManager.java +++ /dev/null @@ -1,1161 +0,0 @@ -/** - * 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.smartdata.metastore.dao.accesscount; - -import org.junit.After; -import org.junit.Assert; -import org.junit.Before; -import org.junit.Test; -import org.smartdata.conf.SmartConf; -import org.smartdata.metastore.MetaStoreException; -import org.smartdata.metastore.dao.Searchable; -import org.smartdata.metastore.dao.TestSearchableDao; -import org.smartdata.metastore.model.AccessCountTable; -import org.smartdata.metastore.model.AggregatedAccessCounts; -import org.smartdata.metastore.queries.PageRequest; -import org.smartdata.metastore.queries.sort.FileAccessInfoSortField; -import org.smartdata.metastore.queries.sort.Sorting; -import org.smartdata.metastore.utils.Constants; -import org.smartdata.metastore.utils.TimeGranularity; -import org.smartdata.metrics.FileAccessEvent; -import org.smartdata.model.FileAccessInfo; -import org.smartdata.model.FileInfo; -import org.smartdata.model.TimeInterval; -import org.smartdata.model.request.FileAccessInfoSearchRequest; - -import java.time.Instant; -import java.util.Arrays; -import java.util.Collections; -import java.util.Comparator; -import java.util.Deque; -import java.util.List; -import java.util.NoSuchElementException; -import java.util.concurrent.ExecutorService; -import java.util.concurrent.Executors; -import java.util.stream.IntStream; - -import static org.junit.Assert.assertEquals; -import static org.junit.Assert.assertTrue; -import static org.smartdata.conf.SmartConfKeys.SMART_ACCESS_COUNT_AGGREGATION_INTERVAL_MS_DEFAULT; -import static org.smartdata.metastore.utils.Constants.ONE_DAY_IN_MILLIS; -import static org.smartdata.metastore.utils.Constants.ONE_HOUR_IN_MILLIS; -import static org.smartdata.metastore.utils.Constants.ONE_MINUTE_IN_MILLIS; -import static org.smartdata.metastore.utils.Constants.ONE_SECOND_IN_MILLIS; - - -public class TestAccessCountTableManager - extends TestSearchableDao { - - private static final List TEST_FILES = Arrays.asList( - "/file0", - "/file1", - "/file2", - "/file3", - "/file4", - "/file5" - ); - - private AccessCountTableManager accessCountTableManager; - private InMemoryAccessCountTableManager inMemoryTableManager; - private ExecutorService executorService; - - @Before - public void setup() { - executorService = Executors.newFixedThreadPool(4); - accessCountTableManager = new AccessCountTableManager( - metaStore, executorService, new SmartConf() - ); - inMemoryTableManager = accessCountTableManager.getInMemoryTableManager(); - } - - @After - public void shutDown() { - executorService.shutdownNow(); - } - - @Override - protected Searchable searchable() { - return accessCountTableManager; - } - - @Override - protected Long getIdentifier(FileAccessInfo fileAccessInfo) { - return fileAccessInfo.getFid(); - } - - @Override - protected FileAccessInfoSortField defaultSortField() { - return FileAccessInfoSortField.FID; - } - - @Test - public void testSearchByFilePath() throws MetaStoreException { - createTestFiles(); - - List accessEvents = Arrays.asList( - new FileAccessEvent("/file1", 1), - new FileAccessEvent("/file2", 3), - new FileAccessEvent("/file4", 4), - new FileAccessEvent("/file2", 5), - new FileAccessEvent("", 5001) - ); - accessCountTableManager.getAccessEventAggregator().aggregate(accessEvents); - - FileAccessInfoSearchRequest searchRequest = FileAccessInfoSearchRequest.builder() - .pathLike("/file") - .build(); - - testSearch(searchRequest, 1L, 2L, 4L); - - searchRequest = FileAccessInfoSearchRequest.builder() - .pathLike("/file2") - .build(); - - testSearch(searchRequest, 2L); - - searchRequest = FileAccessInfoSearchRequest.builder() - .pathLike("another_path") - .build(); - - testSearch(searchRequest); - } - - @Test - public void testCreateTable() throws Exception { - long firstDayEnd = 24 * 60 * 60 * 1000L; - AccessCountTable accessCountTable = - new AccessCountTable(firstDayEnd - 5 * 1000, firstDayEnd); - accessCountTableManager.createTable(accessCountTable); - - Thread.sleep(5000); - - Deque second = - inMemoryTableManager.getTablesOfGranularity(TimeGranularity.SECOND); - - assertEquals(1, second.size()); - assertEquals(accessCountTable, second.peek()); - - Deque minute = - inMemoryTableManager.getTablesOfGranularity(TimeGranularity.MINUTE); - AccessCountTable minuteTable = - new AccessCountTable(firstDayEnd - 60 * 1000, firstDayEnd); - assertEquals(1, minute.size()); - assertEquals(minuteTable, minute.peek()); - - Deque hour = - inMemoryTableManager.getTablesOfGranularity(TimeGranularity.HOUR); - AccessCountTable hourTable = - new AccessCountTable(firstDayEnd - 60 * 60 * 1000, firstDayEnd); - assertEquals(1, hour.size()); - assertEquals(hourTable, hour.peek()); - - Deque day = - inMemoryTableManager.getTablesOfGranularity(TimeGranularity.DAY); - AccessCountTable dayTable = new AccessCountTable(0, firstDayEnd); - assertEquals(1, day.size()); - assertEquals(dayTable, day.peek()); - } - - @Test - public void testGetTables() throws MetaStoreException { - AccessCountTable firstDay = new AccessCountTable(0L, Constants.ONE_DAY_IN_MILLIS); - - AccessCountTable firstHour = - new AccessCountTable(23 * Constants.ONE_HOUR_IN_MILLIS, - 24 * Constants.ONE_HOUR_IN_MILLIS); - AccessCountTable secondHour = - new AccessCountTable(24 * Constants.ONE_HOUR_IN_MILLIS, - 25 * Constants.ONE_HOUR_IN_MILLIS); - - int numMins = 25 * 60; - AccessCountTable firstMin = - new AccessCountTable( - (numMins - 1) * ONE_MINUTE_IN_MILLIS, - numMins * ONE_MINUTE_IN_MILLIS); - AccessCountTable secondMin = - new AccessCountTable( - numMins * ONE_MINUTE_IN_MILLIS, - (numMins + 1) * ONE_MINUTE_IN_MILLIS); - - int numSeconds = (25 * 60 + 1) * 60; - AccessCountTable firstFiveSeconds = - new AccessCountTable( - (numSeconds - 5) * Constants.ONE_SECOND_IN_MILLIS, - numSeconds * Constants.ONE_SECOND_IN_MILLIS); - AccessCountTable secondFiveSeconds = - new AccessCountTable( - numSeconds * Constants.ONE_SECOND_IN_MILLIS, - (numSeconds + 5) * Constants.ONE_SECOND_IN_MILLIS); - - accessCountTableManager.getDbTableManager().createTable(firstDay); - - List tablesToRecover = Arrays.asList( - firstDay, firstHour, secondHour, firstMin, secondMin, firstFiveSeconds, secondFiveSeconds); - inMemoryTableManager.recoverTables(tablesToRecover); - /* - |------------------------interval----------------------------| - |-s-|-s-| - |---m---|---m---| - |-------h-------|-------h------| - |---------------d-----------------| - */ - List firstResult = - accessCountTableManager.getTablesForLast( - (numSeconds + 5) * Constants.ONE_SECOND_IN_MILLIS); - assertEquals(4, firstResult.size()); - assertEquals(firstDay, firstResult.get(0)); - assertEquals(secondHour, firstResult.get(1)); - assertEquals(secondMin, firstResult.get(2)); - assertEquals(secondFiveSeconds, firstResult.get(3)); - - /* - |--------------------interval----------------------------| - |-s-|-s-| - |---m---|---m---| - |-------h-------|-------h------| - |---------------d-----------------| - */ - List secondResult = - accessCountTableManager.getTablesForLast( - numSeconds * Constants.ONE_SECOND_IN_MILLIS); - assertEquals(5, secondResult.size()); - - AccessCountTable firstTable = secondResult.get(0); - assertTrue(firstTable.getStartTime() == 5 * Constants.ONE_SECOND_IN_MILLIS - && firstTable.getEndTime() == 23 * Constants.ONE_HOUR_IN_MILLIS); - - /* - |--------------interval--------------------| - |-s-|-s-| - |---m---|---m---| - |-------h-------|-------h------| - |---------------d-----------------| - */ - List thirdResult = - accessCountTableManager.getTablesForLast( - secondFiveSeconds.getEndTime() - 23 * Constants.ONE_HOUR_IN_MILLIS); - assertEquals(4, thirdResult.size()); - assertEquals(firstHour, thirdResult.get(0)); - - /* - |--------interval----------| - |-s-|-s-| - |---m---|---m---| - |-------h-------|-------h------| - |---------------d-----------------| - */ - List fourthResult = - accessCountTableManager.getTablesForLast( - secondFiveSeconds.getEndTime() - 24 * Constants.ONE_HOUR_IN_MILLIS); - assertEquals(3, fourthResult.size()); - assertEquals(secondHour, fourthResult.get(0)); - } - - @Test - public void testGetTablesCornerCase() throws MetaStoreException { - AccessCountTable firstFiveSeconds = - new AccessCountTable(0L, 5 * Constants.ONE_SECOND_IN_MILLIS); - AccessCountTable secondFiveSeconds = - new AccessCountTable(5 * Constants.ONE_SECOND_IN_MILLIS, - 10 * Constants.ONE_SECOND_IN_MILLIS); - - List tablesToRecover = - Arrays.asList(firstFiveSeconds, secondFiveSeconds); - inMemoryTableManager.recoverTables(tablesToRecover); - - List result = accessCountTableManager.getTablesForLast( - 2 * ONE_MINUTE_IN_MILLIS); - assertEquals(2, result.size()); - assertEquals(firstFiveSeconds, result.get(0)); - assertEquals(secondFiveSeconds, result.get(1)); - } - - @Test - public void testGetTablesCornerCaseMinutes() throws MetaStoreException { - /* - |--interval-| - |-s-|-s-|-s-| - |----m-----| - */ - AccessCountTable firstMinute = - new AccessCountTable(0L, ONE_MINUTE_IN_MILLIS); - AccessCountTable firstFiveSeconds = - new AccessCountTable( - 55 * Constants.ONE_SECOND_IN_MILLIS, 60 * Constants.ONE_SECOND_IN_MILLIS); - AccessCountTable secondFiveSeconds = - new AccessCountTable(60 * Constants.ONE_SECOND_IN_MILLIS, - 65 * Constants.ONE_SECOND_IN_MILLIS); - AccessCountTable thirdFiveSeconds = - new AccessCountTable(110 * Constants.ONE_SECOND_IN_MILLIS, - 115 * Constants.ONE_SECOND_IN_MILLIS); - - List tablesToRecover = Arrays.asList( - firstMinute, firstFiveSeconds, secondFiveSeconds, thirdFiveSeconds); - - inMemoryTableManager.recoverTables(tablesToRecover); - - List result = accessCountTableManager.getTablesForLast( - ONE_MINUTE_IN_MILLIS); - - assertEquals(3, result.size()); - assertEquals(firstFiveSeconds, result.get(0)); - Assert.assertFalse(result.get(0).isEphemeral()); - assertEquals(secondFiveSeconds, result.get(1)); - assertEquals(thirdFiveSeconds, result.get(2)); - } - - @Test - public void testGetAllHotFiles() throws MetaStoreException { - createTestFiles(); - /* - create access count tables for seconds, day and hours intervals - |-------------------------------------interval-----------------------------------------------| - |-s-| |-s-| |-s-| |-s-| |-s-| |-s-| |-s-| |-s-| |-s-| |-s-| |-s-| |-s-| - |---h---|---h----| |---h---|---h---| - |-----------------------d-----------------| - */ - DbAccessCountTableManager tableManager = accessCountTableManager.getDbTableManager(); - AccessCountTable t1 = new AccessCountTable(0, 5 * ONE_SECOND_IN_MILLIS); - AccessCountTable t2 = new AccessCountTable(2 * ONE_MINUTE_IN_MILLIS, - 2 * ONE_MINUTE_IN_MILLIS + 5 * ONE_SECOND_IN_MILLIS); - AccessCountTable t3 = new AccessCountTable(4 * ONE_MINUTE_IN_MILLIS + 10 * ONE_SECOND_IN_MILLIS, - 4 * ONE_MINUTE_IN_MILLIS + 15 * ONE_SECOND_IN_MILLIS); - AccessCountTable t4 = new AccessCountTable(ONE_HOUR_IN_MILLIS + 5 * ONE_MINUTE_IN_MILLIS, - ONE_HOUR_IN_MILLIS + 5 * ONE_MINUTE_IN_MILLIS + 5 * ONE_SECOND_IN_MILLIS); - AccessCountTable t5 = new AccessCountTable(ONE_HOUR_IN_MILLIS + 9 * ONE_MINUTE_IN_MILLIS, - ONE_HOUR_IN_MILLIS + 9 * ONE_MINUTE_IN_MILLIS + 5 * ONE_SECOND_IN_MILLIS); - AccessCountTable t6 = new AccessCountTable(8 * ONE_HOUR_IN_MILLIS, - 8 * ONE_HOUR_IN_MILLIS + 5 * ONE_SECOND_IN_MILLIS); - AccessCountTable t7 = new AccessCountTable(ONE_DAY_IN_MILLIS + 12 * ONE_HOUR_IN_MILLIS, - ONE_DAY_IN_MILLIS + 12 * ONE_HOUR_IN_MILLIS + 5 * ONE_SECOND_IN_MILLIS); - AccessCountTable t8 = new AccessCountTable( - ONE_DAY_IN_MILLIS + 13 * ONE_HOUR_IN_MILLIS + 10 * ONE_SECOND_IN_MILLIS, - ONE_DAY_IN_MILLIS + 13 * ONE_HOUR_IN_MILLIS + 15 * ONE_SECOND_IN_MILLIS); - AccessCountTable t9 = new AccessCountTable(ONE_DAY_IN_MILLIS + 17 * ONE_HOUR_IN_MILLIS, - ONE_DAY_IN_MILLIS + 17 * ONE_HOUR_IN_MILLIS + 5 * ONE_SECOND_IN_MILLIS); - AccessCountTable t10 = new AccessCountTable( - ONE_DAY_IN_MILLIS + 18 * ONE_HOUR_IN_MILLIS + 10 * ONE_SECOND_IN_MILLIS, - ONE_DAY_IN_MILLIS + 18 * ONE_HOUR_IN_MILLIS + 15 * ONE_SECOND_IN_MILLIS); - AccessCountTable t11 = - new AccessCountTable(ONE_DAY_IN_MILLIS + 23 * ONE_HOUR_IN_MILLIS + 59 * ONE_MINUTE_IN_MILLIS - + 58 * ONE_SECOND_IN_MILLIS, 2 * ONE_DAY_IN_MILLIS); - AccessCountTable t12 = new AccessCountTable(0, ONE_DAY_IN_MILLIS); - AccessCountTable t13 = new AccessCountTable(ONE_DAY_IN_MILLIS + 12 * ONE_HOUR_IN_MILLIS, - ONE_DAY_IN_MILLIS + 13 * ONE_HOUR_IN_MILLIS); - AccessCountTable t14 = new AccessCountTable(ONE_DAY_IN_MILLIS + 13 * ONE_HOUR_IN_MILLIS, - ONE_DAY_IN_MILLIS + 14 * ONE_HOUR_IN_MILLIS); - AccessCountTable t15 = new AccessCountTable(ONE_DAY_IN_MILLIS + 17 * ONE_HOUR_IN_MILLIS, - ONE_DAY_IN_MILLIS + 18 * ONE_HOUR_IN_MILLIS); - AccessCountTable t16 = new AccessCountTable(ONE_DAY_IN_MILLIS + 18 * ONE_HOUR_IN_MILLIS, - ONE_DAY_IN_MILLIS + 19 * ONE_HOUR_IN_MILLIS); - - List tables = - Arrays.asList(t1, t2, t3, t4, t5, t6, t7, t8, t9, t10, t11, t12, t13, t14, t15, t16); - - for (AccessCountTable t : tables) { - tableManager.createTable(t); - } - tableManager.handleAggregatedEvents(t1, - Arrays.asList(new AggregatedAccessCounts(0, 1, 0), - new AggregatedAccessCounts(1, 1, 1))); - tableManager.handleAggregatedEvents(t2, - Arrays.asList(new AggregatedAccessCounts(2, 1, 2 * ONE_MINUTE_IN_MILLIS + 1))); - tableManager.handleAggregatedEvents(t3, Arrays.asList( - new AggregatedAccessCounts(1, 1, 4 * ONE_MINUTE_IN_MILLIS + 10 * ONE_SECOND_IN_MILLIS))); - tableManager.handleAggregatedEvents(t4, Arrays.asList( - new AggregatedAccessCounts(2, 1, ONE_HOUR_IN_MILLIS + 5 * ONE_MINUTE_IN_MILLIS))); - tableManager.handleAggregatedEvents(t5, Arrays.asList( - new AggregatedAccessCounts(3, 1, ONE_HOUR_IN_MILLIS + 9 * ONE_MINUTE_IN_MILLIS))); - tableManager.handleAggregatedEvents(t6, - Arrays.asList(new AggregatedAccessCounts(3, 1, 8 * ONE_HOUR_IN_MILLIS))); - tableManager.handleAggregatedEvents(t7, Arrays.asList( - new AggregatedAccessCounts(3, 1, ONE_DAY_IN_MILLIS + 12 * ONE_HOUR_IN_MILLIS))); - tableManager.handleAggregatedEvents(t7, Arrays.asList( - new AggregatedAccessCounts(2, 1, ONE_DAY_IN_MILLIS + 12 * ONE_HOUR_IN_MILLIS + 1))); - tableManager.handleAggregatedEvents(t8, Arrays.asList(new AggregatedAccessCounts(1, 1, - ONE_DAY_IN_MILLIS + 13 * ONE_HOUR_IN_MILLIS + 10 * ONE_SECOND_IN_MILLIS))); - tableManager.handleAggregatedEvents(t9, Arrays.asList( - new AggregatedAccessCounts(4, 1, ONE_DAY_IN_MILLIS + 17 * ONE_HOUR_IN_MILLIS))); - tableManager.handleAggregatedEvents(t10, Arrays.asList( - new AggregatedAccessCounts(5, 1, ONE_DAY_IN_MILLIS + 18 * ONE_HOUR_IN_MILLIS + 10))); - tableManager.handleAggregatedEvents(t11, Arrays.asList( - new AggregatedAccessCounts(3, 1, - ONE_DAY_IN_MILLIS + 23 * ONE_HOUR_IN_MILLIS + 59 * ONE_MINUTE_IN_MILLIS - + 58 * ONE_SECOND_IN_MILLIS))); - tableManager.handleAggregatedEvents(t11, Arrays.asList( - new AggregatedAccessCounts(3, 1, - ONE_DAY_IN_MILLIS + 23 * ONE_HOUR_IN_MILLIS + 59 * ONE_MINUTE_IN_MILLIS - + 59 * ONE_SECOND_IN_MILLIS))); - tableManager.handleAggregatedEvents(t12, - Arrays.asList( - new AggregatedAccessCounts(0, 1, 0), - new AggregatedAccessCounts(1, 2, 4 * ONE_MINUTE_IN_MILLIS + 10 * ONE_SECOND_IN_MILLIS), - new AggregatedAccessCounts(2, 2, ONE_HOUR_IN_MILLIS + 5 * ONE_MINUTE_IN_MILLIS), - new AggregatedAccessCounts(3, 2, 8 * ONE_HOUR_IN_MILLIS) - )); - tableManager.handleAggregatedEvents(t13, Arrays.asList( - new AggregatedAccessCounts(3, 1, ONE_DAY_IN_MILLIS + 12 * ONE_HOUR_IN_MILLIS), - new AggregatedAccessCounts(2, 1, ONE_DAY_IN_MILLIS + 12 * ONE_HOUR_IN_MILLIS + 1) - )); - tableManager.handleAggregatedEvents(t14, Arrays.asList(new AggregatedAccessCounts(1, 1, - ONE_DAY_IN_MILLIS + 13 * ONE_HOUR_IN_MILLIS + 10 * ONE_SECOND_IN_MILLIS))); - tableManager.handleAggregatedEvents(t15, Arrays.asList( - new AggregatedAccessCounts(4, 1, ONE_DAY_IN_MILLIS + 17 * ONE_HOUR_IN_MILLIS))); - tableManager.handleAggregatedEvents(t16, Arrays.asList( - new AggregatedAccessCounts(5, 1, ONE_DAY_IN_MILLIS + 18 * ONE_HOUR_IN_MILLIS + 10))); - - inMemoryTableManager.recoverTables(tables); - - List expectedFiles = Arrays.asList( - new FileAccessInfo(0, TEST_FILES.get(0), 1, 0), - new FileAccessInfo(1, TEST_FILES.get(1), 3, - ONE_DAY_IN_MILLIS + 13 * ONE_HOUR_IN_MILLIS + 10 * ONE_SECOND_IN_MILLIS), - new FileAccessInfo(2, TEST_FILES.get(2), 3, - ONE_DAY_IN_MILLIS + 12 * ONE_HOUR_IN_MILLIS + 1), - new FileAccessInfo(3, TEST_FILES.get(3), 5, - ONE_DAY_IN_MILLIS + 23 * ONE_HOUR_IN_MILLIS + 59 * ONE_MINUTE_IN_MILLIS - + 59 * ONE_SECOND_IN_MILLIS), - new FileAccessInfo(4, TEST_FILES.get(4), 1, ONE_DAY_IN_MILLIS - + 17 * ONE_HOUR_IN_MILLIS), - new FileAccessInfo(5, TEST_FILES.get(5), 1, - ONE_DAY_IN_MILLIS + 18 * ONE_HOUR_IN_MILLIS + 10) - ); - - Long latestAccessedTime = expectedFiles.stream() - .min(Comparator.comparing(FileAccessInfo::getLastAccessedTime)) - .map(FileAccessInfo::getLastAccessedTime) - .orElseThrow(NoSuchElementException::new); - - List hotFiles = - accessCountTableManager.search(FileAccessInfoSearchRequest.builder() - .lastAccessedTime(TimeInterval.builder() - .from(Instant.ofEpochMilli(latestAccessedTime)) - .build()) - .build(), - PageRequest.builder() - .addSorting(FileAccessInfoSortField.FID, Sorting.Order.ASC) - .build()).getItems(); - - assertEquals(expectedFiles, hotFiles); - } - - @Test - public void testGetAllHotFilesDuringLastSeconds() throws MetaStoreException { - createTestFiles(); - submitAccessEvents(); - - List expectedFiles = Arrays.asList( - new FileAccessInfo(3, TEST_FILES.get(3), 2, - ONE_DAY_IN_MILLIS + 23 * ONE_HOUR_IN_MILLIS + 59 * ONE_MINUTE_IN_MILLIS - + 59 * ONE_SECOND_IN_MILLIS) - ); - Long latestAccessedTime = expectedFiles.stream() - .max(Comparator.comparing(FileAccessInfo::getLastAccessedTime)) - .map(FileAccessInfo::getLastAccessedTime) - .orElseThrow(NoSuchElementException::new); - - List hotFiles = - accessCountTableManager.search(FileAccessInfoSearchRequest.builder() - .lastAccessedTime(TimeInterval.builder() - .from(Instant.ofEpochMilli(latestAccessedTime - 5 * ONE_SECOND_IN_MILLIS)) - .to(Instant.ofEpochMilli(latestAccessedTime)) - .build()) - .build()); - - assertEquals(expectedFiles, hotFiles); - } - - @Test - public void testGetHotFilesDuringLastHour() throws MetaStoreException { - createTestFiles(); - submitAccessEvents(); - - List expectedFiles = Collections.singletonList( - new FileAccessInfo(3, TEST_FILES.get(3), 2, - ONE_DAY_IN_MILLIS + 23 * ONE_HOUR_IN_MILLIS + 59 * ONE_MINUTE_IN_MILLIS - + 59 * ONE_SECOND_IN_MILLIS) - ); - - Long latestAccessedTime = expectedFiles.stream() - .max(Comparator.comparing(FileAccessInfo::getLastAccessedTime)) - .map(FileAccessInfo::getLastAccessedTime) - .orElseThrow(NoSuchElementException::new); - - List hotFiles = - accessCountTableManager.search(FileAccessInfoSearchRequest.builder() - .lastAccessedTime(TimeInterval.builder() - .from(Instant.ofEpochMilli(latestAccessedTime - ONE_HOUR_IN_MILLIS)) - .to(Instant.ofEpochMilli(latestAccessedTime)) - .build()) - .build()); - - assertEquals(expectedFiles, hotFiles); - } - - @Test - public void testGetHotFilesFromPartialTable() throws MetaStoreException { - createTestFiles(); - submitAccessEvents(); - - List expectedFiles = Collections.singletonList( - new FileAccessInfo(3, TEST_FILES.get(3), 1, - ONE_DAY_IN_MILLIS + 23 * ONE_HOUR_IN_MILLIS + 59 * ONE_MINUTE_IN_MILLIS - + 59 * ONE_SECOND_IN_MILLIS) - ); - - Long latestAccessedTime = expectedFiles.stream() - .max(Comparator.comparing(FileAccessInfo::getLastAccessedTime)) - .map(FileAccessInfo::getLastAccessedTime) - .orElseThrow(NoSuchElementException::new); - - List hotFiles = - accessCountTableManager.search(FileAccessInfoSearchRequest.builder() - .lastAccessedTime(TimeInterval.builder() - .from(Instant.ofEpochMilli( - (latestAccessedTime - SMART_ACCESS_COUNT_AGGREGATION_INTERVAL_MS_DEFAULT / 2))) - .to(Instant.ofEpochMilli(latestAccessedTime)) - .build()) - .build()); - - assertEquals(expectedFiles, hotFiles); - } - - @Test - public void testGetHotFilesWhenIntervalBiggerThanExistedTables() - throws MetaStoreException { - /* - |---------interval-------------| - |-s-| |-s-| |-s-| - |--------min--------------| - */ - createTestFiles(); - DbAccessCountTableManager tableManager = accessCountTableManager.getDbTableManager(); - AccessCountTable t1 = new AccessCountTable(0, 5 * ONE_SECOND_IN_MILLIS); - AccessCountTable t2 = new AccessCountTable(ONE_MINUTE_IN_MILLIS + 5 * ONE_SECOND_IN_MILLIS, - ONE_MINUTE_IN_MILLIS + 10 * ONE_SECOND_IN_MILLIS); - AccessCountTable t3 = new AccessCountTable(ONE_MINUTE_IN_MILLIS + 20 * ONE_SECOND_IN_MILLIS, - ONE_MINUTE_IN_MILLIS + 25 * ONE_SECOND_IN_MILLIS); - AccessCountTable t4 = new AccessCountTable(ONE_MINUTE_IN_MILLIS + 35 * ONE_SECOND_IN_MILLIS, - ONE_MINUTE_IN_MILLIS + 40 * ONE_SECOND_IN_MILLIS); - AccessCountTable t5 = new AccessCountTable(ONE_MINUTE_IN_MILLIS + 45 * ONE_SECOND_IN_MILLIS, - ONE_MINUTE_IN_MILLIS + 50 * ONE_SECOND_IN_MILLIS); - AccessCountTable t6 = new AccessCountTable(ONE_MINUTE_IN_MILLIS, 2 * ONE_MINUTE_IN_MILLIS); - List tables = Arrays.asList(t1, t2, t3, t4, t5, t6); - tables.forEach(t -> { - try { - tableManager.createTable(t); - } catch (MetaStoreException e) { - throw new RuntimeException(e); - } - }); - tableManager.handleAggregatedEvents(t1, Arrays.asList(new AggregatedAccessCounts(3, 1, 0))); - tableManager.handleAggregatedEvents(t2, Arrays.asList( - new AggregatedAccessCounts(3, 1, ONE_MINUTE_IN_MILLIS + 10 * ONE_SECOND_IN_MILLIS))); - tableManager.handleAggregatedEvents(t3, Arrays.asList( - new AggregatedAccessCounts(2, 1, ONE_MINUTE_IN_MILLIS + 21 * ONE_SECOND_IN_MILLIS), - new AggregatedAccessCounts(4, 1, ONE_MINUTE_IN_MILLIS + 21 * ONE_SECOND_IN_MILLIS))); - tableManager.handleAggregatedEvents(t4, Arrays.asList( - new AggregatedAccessCounts(5, 1, ONE_MINUTE_IN_MILLIS + 38 * ONE_SECOND_IN_MILLIS))); - tableManager.handleAggregatedEvents(t5, Arrays.asList( - new AggregatedAccessCounts(1, 2, ONE_MINUTE_IN_MILLIS + 49 * ONE_SECOND_IN_MILLIS), - new AggregatedAccessCounts(3, 2, ONE_MINUTE_IN_MILLIS + 49 * ONE_SECOND_IN_MILLIS), - new AggregatedAccessCounts(2, 1, ONE_MINUTE_IN_MILLIS + 46 * ONE_SECOND_IN_MILLIS), - new AggregatedAccessCounts(4, 1, ONE_MINUTE_IN_MILLIS + 46 * ONE_SECOND_IN_MILLIS), - new AggregatedAccessCounts(5, 1, ONE_MINUTE_IN_MILLIS + 46 * ONE_SECOND_IN_MILLIS) - )); - tableManager.handleAggregatedEvents(t6, Arrays.asList( - new AggregatedAccessCounts(1, 4, ONE_MINUTE_IN_MILLIS + 49 * ONE_SECOND_IN_MILLIS), - new AggregatedAccessCounts(2, 3, ONE_MINUTE_IN_MILLIS + 49 * ONE_SECOND_IN_MILLIS), - new AggregatedAccessCounts(3, 3, ONE_MINUTE_IN_MILLIS + 49 * ONE_SECOND_IN_MILLIS), - new AggregatedAccessCounts(4, 2, ONE_MINUTE_IN_MILLIS + 46 * ONE_SECOND_IN_MILLIS), - new AggregatedAccessCounts(5, 2, ONE_MINUTE_IN_MILLIS + 46 * ONE_SECOND_IN_MILLIS) - )); - inMemoryTableManager.recoverTables(tables); - - List expectedFiles = Arrays.asList( - new FileAccessInfo(1, TEST_FILES.get(1), 4, - ONE_MINUTE_IN_MILLIS + 49 * ONE_SECOND_IN_MILLIS), - new FileAccessInfo(2, TEST_FILES.get(2), 3, - ONE_MINUTE_IN_MILLIS + 49 * ONE_SECOND_IN_MILLIS), - new FileAccessInfo(3, TEST_FILES.get(3), 3, - ONE_MINUTE_IN_MILLIS + 49 * ONE_SECOND_IN_MILLIS), - new FileAccessInfo(4, TEST_FILES.get(4), 2, - ONE_MINUTE_IN_MILLIS + 46 * ONE_SECOND_IN_MILLIS), - new FileAccessInfo(5, TEST_FILES.get(5), 2, - ONE_MINUTE_IN_MILLIS + 46 * ONE_SECOND_IN_MILLIS) - ); - - List hotFiles = - accessCountTableManager.search(FileAccessInfoSearchRequest.builder() - .lastAccessedTime(TimeInterval.builder() - .from(Instant.ofEpochMilli(55 * ONE_SECOND_IN_MILLIS)) - .to(Instant.ofEpochMilli(2 * ONE_MINUTE_IN_MILLIS + 5 * ONE_SECOND_IN_MILLIS)) - .build()) - .build()); - assertEquals(expectedFiles, hotFiles); - } - - @Test - public void testGetHotFilesWhenIntervalEndTimeLessThanRightBorderOfParentTable() - throws MetaStoreException { - /* - |---------interval------| - |-s-| |-s-| |-s-| - |--------min--------------| - */ - createTestFiles(); - DbAccessCountTableManager tableManager = accessCountTableManager.getDbTableManager(); - AccessCountTable t1 = new AccessCountTable(0, 5 * ONE_SECOND_IN_MILLIS); - AccessCountTable t2 = new AccessCountTable(ONE_MINUTE_IN_MILLIS + 5 * ONE_SECOND_IN_MILLIS, - ONE_MINUTE_IN_MILLIS + 10 * ONE_SECOND_IN_MILLIS); - AccessCountTable t3 = new AccessCountTable(ONE_MINUTE_IN_MILLIS + 20 * ONE_SECOND_IN_MILLIS, - ONE_MINUTE_IN_MILLIS + 25 * ONE_SECOND_IN_MILLIS); - AccessCountTable t4 = new AccessCountTable(ONE_MINUTE_IN_MILLIS + 35 * ONE_SECOND_IN_MILLIS, - ONE_MINUTE_IN_MILLIS + 40 * ONE_SECOND_IN_MILLIS); - AccessCountTable t5 = new AccessCountTable(ONE_MINUTE_IN_MILLIS + 45 * ONE_SECOND_IN_MILLIS, - ONE_MINUTE_IN_MILLIS + 50 * ONE_SECOND_IN_MILLIS); - AccessCountTable t6 = new AccessCountTable(ONE_MINUTE_IN_MILLIS, 2 * ONE_MINUTE_IN_MILLIS); - List tables = Arrays.asList(t1, t2, t3, t4, t5, t6); - tables.forEach(t -> { - try { - tableManager.createTable(t); - } catch (MetaStoreException e) { - throw new RuntimeException(e); - } - }); - tableManager.handleAggregatedEvents(t1, Arrays.asList(new AggregatedAccessCounts(3, 1, 0))); - tableManager.handleAggregatedEvents(t2, Arrays.asList( - new AggregatedAccessCounts(3, 1, ONE_MINUTE_IN_MILLIS + 10 * ONE_SECOND_IN_MILLIS))); - tableManager.handleAggregatedEvents(t3, Arrays.asList( - new AggregatedAccessCounts(2, 1, ONE_MINUTE_IN_MILLIS + 21 * ONE_SECOND_IN_MILLIS), - new AggregatedAccessCounts(4, 1, ONE_MINUTE_IN_MILLIS + 21 * ONE_SECOND_IN_MILLIS))); - tableManager.handleAggregatedEvents(t4, Arrays.asList( - new AggregatedAccessCounts(5, 1, ONE_MINUTE_IN_MILLIS + 38 * ONE_SECOND_IN_MILLIS))); - tableManager.handleAggregatedEvents(t5, Arrays.asList( - new AggregatedAccessCounts(1, 2, ONE_MINUTE_IN_MILLIS + 49 * ONE_SECOND_IN_MILLIS), - new AggregatedAccessCounts(3, 2, ONE_MINUTE_IN_MILLIS + 49 * ONE_SECOND_IN_MILLIS), - new AggregatedAccessCounts(2, 1, ONE_MINUTE_IN_MILLIS + 46 * ONE_SECOND_IN_MILLIS), - new AggregatedAccessCounts(4, 1, ONE_MINUTE_IN_MILLIS + 46 * ONE_SECOND_IN_MILLIS), - new AggregatedAccessCounts(5, 1, ONE_MINUTE_IN_MILLIS + 46 * ONE_SECOND_IN_MILLIS) - )); - tableManager.handleAggregatedEvents(t6, Arrays.asList( - new AggregatedAccessCounts(1, 4, ONE_MINUTE_IN_MILLIS + 49 * ONE_SECOND_IN_MILLIS), - new AggregatedAccessCounts(2, 3, ONE_MINUTE_IN_MILLIS + 49 * ONE_SECOND_IN_MILLIS), - new AggregatedAccessCounts(3, 3, ONE_MINUTE_IN_MILLIS + 49 * ONE_SECOND_IN_MILLIS), - new AggregatedAccessCounts(4, 2, ONE_MINUTE_IN_MILLIS + 46 * ONE_SECOND_IN_MILLIS), - new AggregatedAccessCounts(5, 2, ONE_MINUTE_IN_MILLIS + 46 * ONE_SECOND_IN_MILLIS) - )); - inMemoryTableManager.recoverTables(tables); - - List expectedFiles = Arrays.asList( - new FileAccessInfo(1, TEST_FILES.get(1), 4, - ONE_MINUTE_IN_MILLIS + 49 * ONE_SECOND_IN_MILLIS), - new FileAccessInfo(2, TEST_FILES.get(2), 3, - ONE_MINUTE_IN_MILLIS + 49 * ONE_SECOND_IN_MILLIS), - new FileAccessInfo(3, TEST_FILES.get(3), 3, - ONE_MINUTE_IN_MILLIS + 49 * ONE_SECOND_IN_MILLIS), - new FileAccessInfo(4, TEST_FILES.get(4), 2, - ONE_MINUTE_IN_MILLIS + 46 * ONE_SECOND_IN_MILLIS), - new FileAccessInfo(5, TEST_FILES.get(5), 2, - ONE_MINUTE_IN_MILLIS + 46 * ONE_SECOND_IN_MILLIS) - ); - List hotFiles = - accessCountTableManager.search(FileAccessInfoSearchRequest.builder() - .lastAccessedTime(TimeInterval.builder() - .from(Instant.ofEpochMilli(55 * ONE_SECOND_IN_MILLIS)) - .to(Instant.ofEpochMilli(ONE_MINUTE_IN_MILLIS + 56 * ONE_SECOND_IN_MILLIS)) - .build()) - .build()); - assertEquals(expectedFiles, hotFiles); - } - - @Test - public void testGetHotFilesWhenIntervalStartTimeIncludesIntoChildTable() - throws MetaStoreException { - /* - |--interval-| - |-s-| |-s-| |-s-| - |--------min-------------------| - */ - createTestFiles(); - DbAccessCountTableManager tableManager = accessCountTableManager.getDbTableManager(); - AccessCountTable t4 = new AccessCountTable(ONE_MINUTE_IN_MILLIS + 35 * ONE_SECOND_IN_MILLIS, - ONE_MINUTE_IN_MILLIS + 40 * ONE_SECOND_IN_MILLIS); - AccessCountTable t5 = new AccessCountTable(ONE_MINUTE_IN_MILLIS + 45 * ONE_SECOND_IN_MILLIS, - ONE_MINUTE_IN_MILLIS + 50 * ONE_SECOND_IN_MILLIS); - AccessCountTable t6 = new AccessCountTable(ONE_MINUTE_IN_MILLIS, 2 * ONE_MINUTE_IN_MILLIS); - List tables = Arrays.asList(t4, t5, t6); - tables.forEach(t -> { - try { - tableManager.createTable(t); - } catch (MetaStoreException e) { - throw new RuntimeException(e); - } - }); - tableManager.handleAggregatedEvents(t4, Arrays.asList( - new AggregatedAccessCounts(5, 1, ONE_MINUTE_IN_MILLIS + 38 * ONE_SECOND_IN_MILLIS))); - tableManager.handleAggregatedEvents(t5, Arrays.asList( - new AggregatedAccessCounts(1, 4, ONE_MINUTE_IN_MILLIS + 48 * ONE_SECOND_IN_MILLIS), - new AggregatedAccessCounts(2, 3, ONE_MINUTE_IN_MILLIS + 49 * ONE_SECOND_IN_MILLIS), - new AggregatedAccessCounts(3, 3, ONE_MINUTE_IN_MILLIS + 48 * ONE_SECOND_IN_MILLIS), - new AggregatedAccessCounts(4, 3, ONE_MINUTE_IN_MILLIS + 49 * ONE_SECOND_IN_MILLIS), - new AggregatedAccessCounts(5, 2, ONE_MINUTE_IN_MILLIS + 46 * ONE_SECOND_IN_MILLIS) - )); - tableManager.handleAggregatedEvents(t6, Arrays.asList( - new AggregatedAccessCounts(1, 4, ONE_MINUTE_IN_MILLIS + 48 * ONE_SECOND_IN_MILLIS), - new AggregatedAccessCounts(2, 3, ONE_MINUTE_IN_MILLIS + 49 * ONE_SECOND_IN_MILLIS), - new AggregatedAccessCounts(3, 3, ONE_MINUTE_IN_MILLIS + 48 * ONE_SECOND_IN_MILLIS), - new AggregatedAccessCounts(4, 3, ONE_MINUTE_IN_MILLIS + 49 * ONE_SECOND_IN_MILLIS), - new AggregatedAccessCounts(5, 3, ONE_MINUTE_IN_MILLIS + 46 * ONE_SECOND_IN_MILLIS) - )); - inMemoryTableManager.recoverTables(tables); - - List expectedFiles = Arrays.asList( - new FileAccessInfo(1, TEST_FILES.get(1), 2, - ONE_MINUTE_IN_MILLIS + 48 * ONE_SECOND_IN_MILLIS), - new FileAccessInfo(2, TEST_FILES.get(2), 1, - ONE_MINUTE_IN_MILLIS + 49 * ONE_SECOND_IN_MILLIS), - new FileAccessInfo(3, TEST_FILES.get(3), 1, - ONE_MINUTE_IN_MILLIS + 48 * ONE_SECOND_IN_MILLIS), - new FileAccessInfo(4, TEST_FILES.get(4), 1, - ONE_MINUTE_IN_MILLIS + 49 * ONE_SECOND_IN_MILLIS) - ); - - List hotFiles = - accessCountTableManager.search(FileAccessInfoSearchRequest.builder() - .lastAccessedTime(TimeInterval.builder() - .from(Instant.ofEpochMilli(ONE_MINUTE_IN_MILLIS + 48 * ONE_SECOND_IN_MILLIS)) - .to(Instant.ofEpochMilli(ONE_MINUTE_IN_MILLIS + 56 * ONE_SECOND_IN_MILLIS)) - .build()) - .build()); - assertEquals(expectedFiles, hotFiles); - } - - @Test - public void testGetHotFilesWhenIntervalStartAndEndTimeIncludesIntoChildTables() - throws MetaStoreException { - /* - |--interval-| - |-s-| |-s-| |-s-| - |--------min-----------| - */ - createTestFiles(); - DbAccessCountTableManager tableManager = accessCountTableManager.getDbTableManager(); - AccessCountTable t1 = new AccessCountTable(ONE_MINUTE_IN_MILLIS + 5 * ONE_SECOND_IN_MILLIS, - ONE_MINUTE_IN_MILLIS + 10 * ONE_SECOND_IN_MILLIS); - AccessCountTable t2 = new AccessCountTable(ONE_MINUTE_IN_MILLIS + 20 * ONE_SECOND_IN_MILLIS, - ONE_MINUTE_IN_MILLIS + 25 * ONE_SECOND_IN_MILLIS); - AccessCountTable t4 = new AccessCountTable(ONE_MINUTE_IN_MILLIS + 35 * ONE_SECOND_IN_MILLIS, - ONE_MINUTE_IN_MILLIS + 40 * ONE_SECOND_IN_MILLIS); - AccessCountTable t5 = new AccessCountTable(ONE_MINUTE_IN_MILLIS + 45 * ONE_SECOND_IN_MILLIS, - ONE_MINUTE_IN_MILLIS + 50 * ONE_SECOND_IN_MILLIS); - AccessCountTable t6 = new AccessCountTable(ONE_MINUTE_IN_MILLIS, 2 * ONE_MINUTE_IN_MILLIS); - List tables = Arrays.asList(t1, t2, t4, t5, t6); - tables.forEach(t -> { - try { - tableManager.createTable(t); - } catch (MetaStoreException e) { - throw new RuntimeException(e); - } - }); - tableManager.handleAggregatedEvents(t1, Arrays.asList( - new AggregatedAccessCounts(1, 4, ONE_MINUTE_IN_MILLIS + 9 * ONE_SECOND_IN_MILLIS), - new AggregatedAccessCounts(2, 2, ONE_MINUTE_IN_MILLIS + 8 * ONE_SECOND_IN_MILLIS), - new AggregatedAccessCounts(3, 1, ONE_MINUTE_IN_MILLIS + 5 * ONE_SECOND_IN_MILLIS), - new AggregatedAccessCounts(4, 2, ONE_MINUTE_IN_MILLIS + 9 * ONE_SECOND_IN_MILLIS), - new AggregatedAccessCounts(5, 1, ONE_MINUTE_IN_MILLIS + 7 * ONE_SECOND_IN_MILLIS))); - tableManager.handleAggregatedEvents(t2, Arrays.asList( - new AggregatedAccessCounts(2, 1, ONE_MINUTE_IN_MILLIS + 21 * ONE_SECOND_IN_MILLIS), - new AggregatedAccessCounts(4, 1, ONE_MINUTE_IN_MILLIS + 21 * ONE_SECOND_IN_MILLIS))); - tableManager.handleAggregatedEvents(t4, Arrays.asList( - new AggregatedAccessCounts(5, 1, ONE_MINUTE_IN_MILLIS + 38 * ONE_SECOND_IN_MILLIS))); - tableManager.handleAggregatedEvents(t5, Arrays.asList( - new AggregatedAccessCounts(1, 4, ONE_MINUTE_IN_MILLIS + 48 * ONE_SECOND_IN_MILLIS), - new AggregatedAccessCounts(2, 3, ONE_MINUTE_IN_MILLIS + 49 * ONE_SECOND_IN_MILLIS), - new AggregatedAccessCounts(3, 3, ONE_MINUTE_IN_MILLIS + 48 * ONE_SECOND_IN_MILLIS), - new AggregatedAccessCounts(4, 3, ONE_MINUTE_IN_MILLIS + 49 * ONE_SECOND_IN_MILLIS), - new AggregatedAccessCounts(5, 2, ONE_MINUTE_IN_MILLIS + 46 * ONE_SECOND_IN_MILLIS) - )); - tableManager.handleAggregatedEvents(t6, Arrays.asList( - new AggregatedAccessCounts(1, 4, ONE_MINUTE_IN_MILLIS + 48 * ONE_SECOND_IN_MILLIS), - new AggregatedAccessCounts(2, 3, ONE_MINUTE_IN_MILLIS + 49 * ONE_SECOND_IN_MILLIS), - new AggregatedAccessCounts(3, 3, ONE_MINUTE_IN_MILLIS + 48 * ONE_SECOND_IN_MILLIS), - new AggregatedAccessCounts(4, 3, ONE_MINUTE_IN_MILLIS + 49 * ONE_SECOND_IN_MILLIS), - new AggregatedAccessCounts(5, 3, ONE_MINUTE_IN_MILLIS + 46 * ONE_SECOND_IN_MILLIS) - )); - inMemoryTableManager.recoverTables(tables); - - List expectedFiles = Arrays.asList( - new FileAccessInfo(1, TEST_FILES.get(1), 4, - ONE_MINUTE_IN_MILLIS + 48 * ONE_SECOND_IN_MILLIS), - new FileAccessInfo(2, TEST_FILES.get(2), 2, - ONE_MINUTE_IN_MILLIS + 21 * ONE_SECOND_IN_MILLIS), - new FileAccessInfo(3, TEST_FILES.get(3), 2, - ONE_MINUTE_IN_MILLIS + 48 * ONE_SECOND_IN_MILLIS), - new FileAccessInfo(4, TEST_FILES.get(4), 2, - ONE_MINUTE_IN_MILLIS + 21 * ONE_SECOND_IN_MILLIS), - new FileAccessInfo(5, TEST_FILES.get(5), 2, - ONE_MINUTE_IN_MILLIS + 46 * ONE_SECOND_IN_MILLIS) - ); - - List hotFiles = - accessCountTableManager.search(FileAccessInfoSearchRequest.builder() - .lastAccessedTime(TimeInterval.builder() - .from(Instant.ofEpochMilli(ONE_MINUTE_IN_MILLIS + 8 * ONE_SECOND_IN_MILLIS)) - .to(Instant.ofEpochMilli(ONE_MINUTE_IN_MILLIS + 48 * ONE_SECOND_IN_MILLIS)) - .build()) - .build()); - assertEquals(expectedFiles, hotFiles); - } - - @Test - public void testGetHotFilesWhenIntervalBiggerThanExistedTablesCombineGranularity() - throws MetaStoreException { - /* - |---------interval------------------------| - |-s-| |-s-| |-s-| - |-----min-----------|-------------------| - */ - createTestFiles(); - DbAccessCountTableManager tableManager = accessCountTableManager.getDbTableManager(); - - AccessCountTable t0 = new AccessCountTable(0, ONE_MINUTE_IN_MILLIS); - AccessCountTable t1 = new AccessCountTable(ONE_MINUTE_IN_MILLIS + 5 * ONE_SECOND_IN_MILLIS, - ONE_MINUTE_IN_MILLIS + 10 * ONE_SECOND_IN_MILLIS); - AccessCountTable t2 = new AccessCountTable(ONE_MINUTE_IN_MILLIS + 20 * ONE_SECOND_IN_MILLIS, - ONE_MINUTE_IN_MILLIS + 25 * ONE_SECOND_IN_MILLIS); - AccessCountTable t4 = new AccessCountTable(ONE_MINUTE_IN_MILLIS + 35 * ONE_SECOND_IN_MILLIS, - ONE_MINUTE_IN_MILLIS + 40 * ONE_SECOND_IN_MILLIS); - AccessCountTable t5 = new AccessCountTable(ONE_MINUTE_IN_MILLIS + 45 * ONE_SECOND_IN_MILLIS, - ONE_MINUTE_IN_MILLIS + 50 * ONE_SECOND_IN_MILLIS); - AccessCountTable t6 = new AccessCountTable(ONE_MINUTE_IN_MILLIS, 2 * ONE_MINUTE_IN_MILLIS); - List tables = Arrays.asList(t0, t1, t2, t4, t5, t6); - tables.forEach(t -> { - try { - tableManager.createTable(t); - } catch (MetaStoreException e) { - throw new RuntimeException(e); - } - }); - tableManager.handleAggregatedEvents(t0, Arrays.asList( - new AggregatedAccessCounts(1, 2, 38 * ONE_SECOND_IN_MILLIS), - new AggregatedAccessCounts(3, 5, 44 * ONE_SECOND_IN_MILLIS), - new AggregatedAccessCounts(4, 1, 45 * ONE_SECOND_IN_MILLIS), - new AggregatedAccessCounts(5, 2, 51 * ONE_SECOND_IN_MILLIS))); - tableManager.handleAggregatedEvents(t1, Arrays.asList( - new AggregatedAccessCounts(1, 4, ONE_MINUTE_IN_MILLIS + 9 * ONE_SECOND_IN_MILLIS), - new AggregatedAccessCounts(2, 2, ONE_MINUTE_IN_MILLIS + 8 * ONE_SECOND_IN_MILLIS), - new AggregatedAccessCounts(3, 1, ONE_MINUTE_IN_MILLIS + 5 * ONE_SECOND_IN_MILLIS), - new AggregatedAccessCounts(4, 2, ONE_MINUTE_IN_MILLIS + 9 * ONE_SECOND_IN_MILLIS), - new AggregatedAccessCounts(5, 1, ONE_MINUTE_IN_MILLIS + 7 * ONE_SECOND_IN_MILLIS))); - tableManager.handleAggregatedEvents(t2, Arrays.asList( - new AggregatedAccessCounts(2, 1, ONE_MINUTE_IN_MILLIS + 21 * ONE_SECOND_IN_MILLIS), - new AggregatedAccessCounts(4, 1, ONE_MINUTE_IN_MILLIS + 21 * ONE_SECOND_IN_MILLIS))); - tableManager.handleAggregatedEvents(t4, Arrays.asList( - new AggregatedAccessCounts(5, 1, ONE_MINUTE_IN_MILLIS + 38 * ONE_SECOND_IN_MILLIS))); - tableManager.handleAggregatedEvents(t5, Arrays.asList( - new AggregatedAccessCounts(1, 4, ONE_MINUTE_IN_MILLIS + 48 * ONE_SECOND_IN_MILLIS), - new AggregatedAccessCounts(2, 3, ONE_MINUTE_IN_MILLIS + 49 * ONE_SECOND_IN_MILLIS), - new AggregatedAccessCounts(3, 3, ONE_MINUTE_IN_MILLIS + 48 * ONE_SECOND_IN_MILLIS), - new AggregatedAccessCounts(4, 3, ONE_MINUTE_IN_MILLIS + 49 * ONE_SECOND_IN_MILLIS), - new AggregatedAccessCounts(5, 2, ONE_MINUTE_IN_MILLIS + 46 * ONE_SECOND_IN_MILLIS) - )); - tableManager.handleAggregatedEvents(t6, Arrays.asList( - new AggregatedAccessCounts(1, 4, ONE_MINUTE_IN_MILLIS + 48 * ONE_SECOND_IN_MILLIS), - new AggregatedAccessCounts(2, 3, ONE_MINUTE_IN_MILLIS + 49 * ONE_SECOND_IN_MILLIS), - new AggregatedAccessCounts(3, 3, ONE_MINUTE_IN_MILLIS + 48 * ONE_SECOND_IN_MILLIS), - new AggregatedAccessCounts(4, 3, ONE_MINUTE_IN_MILLIS + 49 * ONE_SECOND_IN_MILLIS), - new AggregatedAccessCounts(5, 3, ONE_MINUTE_IN_MILLIS + 46 * ONE_SECOND_IN_MILLIS) - )); - inMemoryTableManager.recoverTables(tables); - - List expectedFiles = Arrays.asList( - new FileAccessInfo(1, TEST_FILES.get(1), 6, - ONE_MINUTE_IN_MILLIS + 48 * ONE_SECOND_IN_MILLIS), - new FileAccessInfo(2, TEST_FILES.get(2), 3, - ONE_MINUTE_IN_MILLIS + 49 * ONE_SECOND_IN_MILLIS), - new FileAccessInfo(3, TEST_FILES.get(3), 8, - ONE_MINUTE_IN_MILLIS + 48 * ONE_SECOND_IN_MILLIS), - new FileAccessInfo(4, TEST_FILES.get(4), 4, - ONE_MINUTE_IN_MILLIS + 49 * ONE_SECOND_IN_MILLIS), - new FileAccessInfo(5, TEST_FILES.get(5), 5, - ONE_MINUTE_IN_MILLIS + 46 * ONE_SECOND_IN_MILLIS) - ); - - List hotFiles = - accessCountTableManager.search(FileAccessInfoSearchRequest.builder() - .lastAccessedTime(TimeInterval.builder() - .from(Instant.ofEpochMilli(0)) - .to(Instant.ofEpochMilli(2 * ONE_MINUTE_IN_MILLIS + 4 * ONE_SECOND_IN_MILLIS)) - .build()) - .build()); - assertEquals(expectedFiles, hotFiles); - } - - @Test - public void testGetHotFilesWhenIntervalEndsWithinSmallestTableCombinedGranularity() - throws MetaStoreException { - /* - |---------interval------------| - |-s-| |-s-| - |-----min----------|--------------| - |-----------------hour--------...-----| - */ - createTestFiles(); - DbAccessCountTableManager tableManager = accessCountTableManager.getDbTableManager(); - AccessCountTable t0 = new AccessCountTable(0, ONE_MINUTE_IN_MILLIS); - AccessCountTable t1 = new AccessCountTable(ONE_MINUTE_IN_MILLIS + 5 * ONE_SECOND_IN_MILLIS, - ONE_MINUTE_IN_MILLIS + 10 * ONE_SECOND_IN_MILLIS); - AccessCountTable t2 = new AccessCountTable(ONE_MINUTE_IN_MILLIS + 20 * ONE_SECOND_IN_MILLIS, - ONE_MINUTE_IN_MILLIS + 25 * ONE_SECOND_IN_MILLIS); - AccessCountTable t4 = new AccessCountTable(ONE_MINUTE_IN_MILLIS + 35 * ONE_SECOND_IN_MILLIS, - ONE_MINUTE_IN_MILLIS + 40 * ONE_SECOND_IN_MILLIS); - AccessCountTable t5 = new AccessCountTable(ONE_MINUTE_IN_MILLIS + 45 * ONE_SECOND_IN_MILLIS, - ONE_MINUTE_IN_MILLIS + 50 * ONE_SECOND_IN_MILLIS); - AccessCountTable t6 = new AccessCountTable(ONE_MINUTE_IN_MILLIS, 2 * ONE_MINUTE_IN_MILLIS); - AccessCountTable t7 = new AccessCountTable(0, ONE_HOUR_IN_MILLIS); - List tables = Arrays.asList(t0, t1, t2, t4, t5, t6, t7); - tables.forEach(t -> { - try { - tableManager.createTable(t); - } catch (MetaStoreException e) { - throw new RuntimeException(e); - } - }); - tableManager.handleAggregatedEvents(t0, Arrays.asList( - new AggregatedAccessCounts(1, 2, 38 * ONE_SECOND_IN_MILLIS), - new AggregatedAccessCounts(3, 5, 44 * ONE_SECOND_IN_MILLIS), - new AggregatedAccessCounts(4, 1, 45 * ONE_SECOND_IN_MILLIS), - new AggregatedAccessCounts(5, 2, 51 * ONE_SECOND_IN_MILLIS))); - tableManager.handleAggregatedEvents(t1, Arrays.asList( - new AggregatedAccessCounts(1, 4, ONE_MINUTE_IN_MILLIS + 9 * ONE_SECOND_IN_MILLIS), - new AggregatedAccessCounts(2, 2, ONE_MINUTE_IN_MILLIS + 8 * ONE_SECOND_IN_MILLIS), - new AggregatedAccessCounts(3, 1, ONE_MINUTE_IN_MILLIS + 5 * ONE_SECOND_IN_MILLIS), - new AggregatedAccessCounts(4, 2, ONE_MINUTE_IN_MILLIS + 9 * ONE_SECOND_IN_MILLIS), - new AggregatedAccessCounts(5, 1, ONE_MINUTE_IN_MILLIS + 7 * ONE_SECOND_IN_MILLIS))); - tableManager.handleAggregatedEvents(t2, Arrays.asList( - new AggregatedAccessCounts(2, 1, ONE_MINUTE_IN_MILLIS + 21 * ONE_SECOND_IN_MILLIS), - new AggregatedAccessCounts(4, 1, ONE_MINUTE_IN_MILLIS + 21 * ONE_SECOND_IN_MILLIS))); - tableManager.handleAggregatedEvents(t4, Arrays.asList( - new AggregatedAccessCounts(5, 1, ONE_MINUTE_IN_MILLIS + 38 * ONE_SECOND_IN_MILLIS))); - tableManager.handleAggregatedEvents(t5, Arrays.asList( - new AggregatedAccessCounts(1, 4, ONE_MINUTE_IN_MILLIS + 48 * ONE_SECOND_IN_MILLIS), - new AggregatedAccessCounts(2, 3, ONE_MINUTE_IN_MILLIS + 49 * ONE_SECOND_IN_MILLIS), - new AggregatedAccessCounts(3, 3, ONE_MINUTE_IN_MILLIS + 48 * ONE_SECOND_IN_MILLIS), - new AggregatedAccessCounts(4, 3, ONE_MINUTE_IN_MILLIS + 49 * ONE_SECOND_IN_MILLIS), - new AggregatedAccessCounts(5, 2, ONE_MINUTE_IN_MILLIS + 46 * ONE_SECOND_IN_MILLIS) - )); - tableManager.handleAggregatedEvents(t6, Arrays.asList( - new AggregatedAccessCounts(1, 4, ONE_MINUTE_IN_MILLIS + 48 * ONE_SECOND_IN_MILLIS), - new AggregatedAccessCounts(2, 3, ONE_MINUTE_IN_MILLIS + 49 * ONE_SECOND_IN_MILLIS), - new AggregatedAccessCounts(3, 3, ONE_MINUTE_IN_MILLIS + 48 * ONE_SECOND_IN_MILLIS), - new AggregatedAccessCounts(4, 3, ONE_MINUTE_IN_MILLIS + 49 * ONE_SECOND_IN_MILLIS), - new AggregatedAccessCounts(5, 3, ONE_MINUTE_IN_MILLIS + 46 * ONE_SECOND_IN_MILLIS) - )); - tableManager.handleAggregatedEvents(t7, Arrays.asList( - new AggregatedAccessCounts(1, 24, 50 * ONE_MINUTE_IN_MILLIS), - new AggregatedAccessCounts(2, 18, 49 * ONE_MINUTE_IN_MILLIS), - new AggregatedAccessCounts(3, 18, 30 * ONE_MINUTE_IN_MILLIS), - new AggregatedAccessCounts(4, 15, 55 * ONE_MINUTE_IN_MILLIS), - new AggregatedAccessCounts(5, 27, 57 * ONE_MINUTE_IN_MILLIS) - )); - inMemoryTableManager.recoverTables(tables); - - List expectedFiles = Arrays.asList( - new FileAccessInfo(1, TEST_FILES.get(1), 8, - ONE_MINUTE_IN_MILLIS + 48 * ONE_SECOND_IN_MILLIS), - new FileAccessInfo(2, TEST_FILES.get(2), 3, - ONE_MINUTE_IN_MILLIS + 21 * ONE_SECOND_IN_MILLIS), - new FileAccessInfo(3, TEST_FILES.get(3), 8, - ONE_MINUTE_IN_MILLIS + 48 * ONE_SECOND_IN_MILLIS), - new FileAccessInfo(4, TEST_FILES.get(4), 4, - ONE_MINUTE_IN_MILLIS + 21 * ONE_SECOND_IN_MILLIS), - new FileAccessInfo(5, TEST_FILES.get(5), 5, - ONE_MINUTE_IN_MILLIS + 46 * ONE_SECOND_IN_MILLIS) - ); - - List hotFiles = - accessCountTableManager.search(FileAccessInfoSearchRequest.builder() - .lastAccessedTime(TimeInterval.builder() - .from(Instant.ofEpochMilli(0)) - .to(Instant.ofEpochMilli(ONE_MINUTE_IN_MILLIS + 48 * ONE_SECOND_IN_MILLIS)) - .build()) - .build()); - assertEquals(expectedFiles, hotFiles); - } - - @Test - public void testGetHotFilesWhenIntervalStartsInParentTableAndEndsInChildTable() - throws MetaStoreException { - /* - |---interval--| - |-s-| - |---------m------------| - */ - createTestFiles(); - DbAccessCountTableManager tableManager = accessCountTableManager.getDbTableManager(); - - AccessCountTable t5 = new AccessCountTable(ONE_MINUTE_IN_MILLIS + 45 * ONE_SECOND_IN_MILLIS, - ONE_MINUTE_IN_MILLIS + 50 * ONE_SECOND_IN_MILLIS); - AccessCountTable t6 = new AccessCountTable(ONE_MINUTE_IN_MILLIS, 2 * ONE_MINUTE_IN_MILLIS); - List tables = Arrays.asList(t5, t6); - tables.forEach(t -> { - try { - tableManager.createTable(t); - } catch (MetaStoreException e) { - throw new RuntimeException(e); - } - }); - tableManager.handleAggregatedEvents(t5, Arrays.asList( - new AggregatedAccessCounts(1, 4, ONE_MINUTE_IN_MILLIS + 48 * ONE_SECOND_IN_MILLIS), - new AggregatedAccessCounts(2, 3, ONE_MINUTE_IN_MILLIS + 49 * ONE_SECOND_IN_MILLIS), - new AggregatedAccessCounts(3, 3, ONE_MINUTE_IN_MILLIS + 48 * ONE_SECOND_IN_MILLIS), - new AggregatedAccessCounts(4, 3, ONE_MINUTE_IN_MILLIS + 49 * ONE_SECOND_IN_MILLIS), - new AggregatedAccessCounts(5, 2, ONE_MINUTE_IN_MILLIS + 46 * ONE_SECOND_IN_MILLIS) - )); - tableManager.handleAggregatedEvents(t6, Arrays.asList( - new AggregatedAccessCounts(1, 10, ONE_MINUTE_IN_MILLIS + 48 * ONE_SECOND_IN_MILLIS), - new AggregatedAccessCounts(2, 5, ONE_MINUTE_IN_MILLIS + 49 * ONE_SECOND_IN_MILLIS), - new AggregatedAccessCounts(3, 7, ONE_MINUTE_IN_MILLIS + 48 * ONE_SECOND_IN_MILLIS), - new AggregatedAccessCounts(4, 8, ONE_MINUTE_IN_MILLIS + 49 * ONE_SECOND_IN_MILLIS), - new AggregatedAccessCounts(5, 4, ONE_MINUTE_IN_MILLIS + 46 * ONE_SECOND_IN_MILLIS) - )); - inMemoryTableManager.recoverTables(tables); - - List expectedFiles = Arrays.asList( - new FileAccessInfo(1, TEST_FILES.get(1), 2, - ONE_MINUTE_IN_MILLIS + 48 * ONE_SECOND_IN_MILLIS), - new FileAccessInfo(3, TEST_FILES.get(3), 2, - ONE_MINUTE_IN_MILLIS + 48 * ONE_SECOND_IN_MILLIS), - new FileAccessInfo(5, TEST_FILES.get(5), 1, - ONE_MINUTE_IN_MILLIS + 46 * ONE_SECOND_IN_MILLIS) - ); - - List hotFiles = - accessCountTableManager.search(FileAccessInfoSearchRequest.builder() - .lastAccessedTime(TimeInterval.builder() - .from(Instant.ofEpochMilli(ONE_MINUTE_IN_MILLIS + 10 * ONE_SECOND_IN_MILLIS)) - .to(Instant.ofEpochMilli(ONE_MINUTE_IN_MILLIS + 48 * ONE_SECOND_IN_MILLIS)) - .build()) - .build()); - assertEquals(expectedFiles, hotFiles); - } - - @Test - public void testGetHotFilesWhenIntervalIncludesOnlyInParentTable() - throws MetaStoreException { - /* - |--interval--| - |-s-| - |---------m------------| - */ - createTestFiles(); - DbAccessCountTableManager tableManager = accessCountTableManager.getDbTableManager(); - - AccessCountTable t5 = new AccessCountTable(ONE_MINUTE_IN_MILLIS + 45 * ONE_SECOND_IN_MILLIS, - ONE_MINUTE_IN_MILLIS + 50 * ONE_SECOND_IN_MILLIS); - AccessCountTable t6 = new AccessCountTable(ONE_MINUTE_IN_MILLIS, 2 * ONE_MINUTE_IN_MILLIS); - List tables = Arrays.asList(t5, t6); - tables.forEach(t -> { - try { - tableManager.createTable(t); - } catch (MetaStoreException e) { - throw new RuntimeException(e); - } - }); - tableManager.handleAggregatedEvents(t5, Arrays.asList( - new AggregatedAccessCounts(1, 4, ONE_MINUTE_IN_MILLIS + 48 * ONE_SECOND_IN_MILLIS), - new AggregatedAccessCounts(3, 3, ONE_MINUTE_IN_MILLIS + 48 * ONE_SECOND_IN_MILLIS), - new AggregatedAccessCounts(4, 3, ONE_MINUTE_IN_MILLIS + 49 * ONE_SECOND_IN_MILLIS) - )); - tableManager.handleAggregatedEvents(t6, Arrays.asList( - new AggregatedAccessCounts(1, 10, ONE_MINUTE_IN_MILLIS + 48 * ONE_SECOND_IN_MILLIS), - new AggregatedAccessCounts(2, 15, ONE_MINUTE_IN_MILLIS + 20 * ONE_SECOND_IN_MILLIS), - new AggregatedAccessCounts(3, 7, ONE_MINUTE_IN_MILLIS + 48 * ONE_SECOND_IN_MILLIS), - new AggregatedAccessCounts(4, 8, ONE_MINUTE_IN_MILLIS + 49 * ONE_SECOND_IN_MILLIS), - new AggregatedAccessCounts(5, 4, ONE_MINUTE_IN_MILLIS + 30 * ONE_SECOND_IN_MILLIS) - )); - inMemoryTableManager.recoverTables(tables); - - List expectedFiles = Arrays.asList( - new FileAccessInfo(2, TEST_FILES.get(2), 8, - ONE_MINUTE_IN_MILLIS + 20 * ONE_SECOND_IN_MILLIS), - new FileAccessInfo(5, TEST_FILES.get(5), 2, - ONE_MINUTE_IN_MILLIS + 30 * ONE_SECOND_IN_MILLIS) - ); - - List hotFiles = - accessCountTableManager.search(FileAccessInfoSearchRequest.builder() - .lastAccessedTime(TimeInterval.builder() - .from(Instant.ofEpochMilli(ONE_MINUTE_IN_MILLIS + 10 * ONE_SECOND_IN_MILLIS)) - .to(Instant.ofEpochMilli(ONE_MINUTE_IN_MILLIS + 40 * ONE_SECOND_IN_MILLIS)) - .build()) - .build()); - assertEquals(expectedFiles, hotFiles); - } - - private void submitAccessEvents() { - InMemoryAccessEventAggregator accessEventAggregator = - accessCountTableManager.getAccessEventAggregator(); - List accessEvents = Arrays.asList( - new FileAccessEvent(TEST_FILES.get(0), 0), - new FileAccessEvent(TEST_FILES.get(1), 1), - new FileAccessEvent(TEST_FILES.get(2), - 2 * ONE_MINUTE_IN_MILLIS + 1), - new FileAccessEvent("/unknown", - 3 * ONE_MINUTE_IN_MILLIS), - new FileAccessEvent(TEST_FILES.get(1), - 4 * ONE_MINUTE_IN_MILLIS + 10 * ONE_SECOND_IN_MILLIS), - new FileAccessEvent(TEST_FILES.get(2), - ONE_HOUR_IN_MILLIS + 5 * ONE_MINUTE_IN_MILLIS), - new FileAccessEvent(TEST_FILES.get(3), - ONE_HOUR_IN_MILLIS + 9 * ONE_MINUTE_IN_MILLIS), - new FileAccessEvent(TEST_FILES.get(3), - 8 * ONE_HOUR_IN_MILLIS), - new FileAccessEvent(TEST_FILES.get(3), - ONE_DAY_IN_MILLIS + 12 * ONE_HOUR_IN_MILLIS), - new FileAccessEvent(TEST_FILES.get(2), - ONE_DAY_IN_MILLIS + 12 * ONE_HOUR_IN_MILLIS + 1), - new FileAccessEvent(TEST_FILES.get(1), - ONE_DAY_IN_MILLIS + 13 * ONE_HOUR_IN_MILLIS + 10 * ONE_SECOND_IN_MILLIS), - new FileAccessEvent(TEST_FILES.get(4), - ONE_DAY_IN_MILLIS + 17 * ONE_HOUR_IN_MILLIS), - new FileAccessEvent(TEST_FILES.get(5), - ONE_DAY_IN_MILLIS + 18 * ONE_HOUR_IN_MILLIS + 10), - new FileAccessEvent(TEST_FILES.get(3), - ONE_DAY_IN_MILLIS + 23 * ONE_HOUR_IN_MILLIS + 59 * ONE_MINUTE_IN_MILLIS - + 58 * ONE_SECOND_IN_MILLIS), - new FileAccessEvent(TEST_FILES.get(3), - ONE_DAY_IN_MILLIS + 23 * ONE_HOUR_IN_MILLIS + 59 * ONE_MINUTE_IN_MILLIS - + 59 * ONE_SECOND_IN_MILLIS), - new FileAccessEvent("", 2 * ONE_DAY_IN_MILLIS) - ); - accessEventAggregator.aggregate(accessEvents); - } - - private void createTestFiles() throws MetaStoreException { - FileInfo[] fileInfos = IntStream.range(0, TEST_FILES.size()) - .mapToObj(id -> FileInfo.newBuilder() - .setFileId(id) - .setPath(TEST_FILES.get(id)) - .build()) - .toArray(FileInfo[]::new); - - metaStore.insertFiles(fileInfos); - } -} diff --git a/smart-metastore/src/test/java/org/smartdata/metastore/dao/accesscount/TestAccessEventAggregator.java b/smart-metastore/src/test/java/org/smartdata/metastore/dao/accesscount/TestAccessEventAggregator.java index e50a504313a..8b5fab634b1 100644 --- a/smart-metastore/src/test/java/org/smartdata/metastore/dao/accesscount/TestAccessEventAggregator.java +++ b/smart-metastore/src/test/java/org/smartdata/metastore/dao/accesscount/TestAccessEventAggregator.java @@ -17,108 +17,85 @@ */ package org.smartdata.metastore.dao.accesscount; -import lombok.Getter; import org.junit.Before; import org.junit.Test; import org.smartdata.metastore.TestDaoBase; -import org.smartdata.metastore.dao.accesscount.InMemoryAccessEventAggregator.WindowClosedCallback; -import org.smartdata.metastore.model.AccessCountTable; -import org.smartdata.metastore.model.AggregatedAccessCounts; +import org.smartdata.metastore.accesscount.DbAccessEventAggregator; +import org.smartdata.metastore.accesscount.FileAccessManager; +import org.smartdata.metastore.accesscount.failover.AccessCountContext; +import org.smartdata.metastore.accesscount.failover.Failover; +import org.smartdata.metastore.model.SearchResult; +import org.smartdata.metastore.queries.PageRequest; +import org.smartdata.metastore.queries.sort.FileAccessInfoSortField; +import org.smartdata.metastore.queries.sort.Sorting; +import org.smartdata.metastore.transaction.TransactionRunner; import org.smartdata.metrics.FileAccessEvent; +import org.smartdata.model.FileAccessInfo; import org.smartdata.model.FileInfo; +import org.smartdata.model.request.FileAccessInfoSearchRequest; import org.testcontainers.shaded.com.google.common.collect.ImmutableMap; -import java.util.ArrayList; import java.util.Arrays; -import java.util.Collection; import java.util.Collections; -import java.util.Comparator; -import java.util.List; import java.util.Map; import static org.junit.Assert.assertEquals; -import static org.junit.Assert.assertTrue; public class TestAccessEventAggregator extends TestDaoBase { - private static final long AGGREGATION_GRANULARITY_MS = 5000; private static final Map FILE_PATH_IDS = ImmutableMap.of( "/file1", 1L, "/file2", 2L, "/file3", 3L, "/file4", 4L ); - - private InMemoryAccessEventAggregator aggregator; - private MockWindowClosedCallback windowCallback; + private FileAccessManager dbTableManager; + private DbAccessEventAggregator aggregator; @Before public void setup() { - windowCallback = new MockWindowClosedCallback(); - aggregator = new InMemoryAccessEventAggregator( - metaStore.fileInfoDao(), - windowCallback, - AccessCountEventAggregatorFailover.dropEvents(), - AGGREGATION_GRANULARITY_MS); - + dbTableManager = + new FileAccessManager(new TransactionRunner(metaStore.transactionManager()), + metaStore.accessCountEventDao(), + metaStore.cacheFileDao()); + aggregator = + new DbAccessEventAggregator(metaStore.fileInfoDao(), + dbTableManager, new Failover(){}); metaStore.fileInfoDao().insert(testFileInfos()); } @Test public void testAggregateEvents() { - List createdTables = windowCallback.getCreatedTables(); - List collectedAccessCounts = - windowCallback.getCollectedAccessCounts(); - - aggregator.aggregate(Collections.singletonList(new FileAccessEvent("", 3000))); - assertTrue(createdTables.isEmpty()); - assertTrue(collectedAccessCounts.isEmpty()); - + long currentTimeMs = System.currentTimeMillis(); + aggregator.aggregate(Collections.singletonList(new FileAccessEvent("", currentTimeMs))); aggregator.aggregate(Collections.singletonList - (new FileAccessEvent("/file1", 4999))); + (new FileAccessEvent("/file1", currentTimeMs + 1))); aggregator.aggregate(Collections.singletonList( - new FileAccessEvent("", 6000))); - - assertEquals( - Collections.singletonList(new AccessCountTable(0, 5000)), - createdTables); - assertEquals( - Collections.singletonList( - new AggregatedAccessCounts(FILE_PATH_IDS.get("/file1"), 1, 4999)), - collectedAccessCounts); - + new FileAccessEvent("", currentTimeMs + 1000))); aggregator.aggregate( Arrays.asList( - new FileAccessEvent("/file1", 7900), - new FileAccessEvent("/file1", 7999), - new FileAccessEvent("/file1", 8000), - new FileAccessEvent("/file2", 14000), - new FileAccessEvent("/file3", 14000), - new FileAccessEvent("/file3", 14001), - new FileAccessEvent("/file3", 14002), - new FileAccessEvent("/unknown_file", 14003), - new FileAccessEvent("/file4", 16000), - new FileAccessEvent("", 22000))); - - List expectedTables = Arrays.asList( - new AccessCountTable(0, 5000), - new AccessCountTable(5000, 10000), - new AccessCountTable(10000, 15000), - new AccessCountTable(15000, 20000) - ); - assertEquals(expectedTables, createdTables); - - List expectedAccessCounts = Arrays.asList( - new AggregatedAccessCounts(FILE_PATH_IDS.get("/file1"), 1, 4999), - new AggregatedAccessCounts(FILE_PATH_IDS.get("/file1"), 3, 8000), - new AggregatedAccessCounts(FILE_PATH_IDS.get("/file2"), 1, 14000), - new AggregatedAccessCounts(FILE_PATH_IDS.get("/file3"), 3, 14002), - new AggregatedAccessCounts(FILE_PATH_IDS.get("/file4"), 1, 16000) - ); - - collectedAccessCounts.sort( - Comparator.comparingLong(AggregatedAccessCounts::getLastAccessedTimestamp)); - assertEquals(expectedAccessCounts, collectedAccessCounts); + new FileAccessEvent("/file1", currentTimeMs + 2000), + new FileAccessEvent("/file1", currentTimeMs + 7999), + new FileAccessEvent("/file1", currentTimeMs + 8000), + new FileAccessEvent("/file2", currentTimeMs + 14000), + new FileAccessEvent("/file3", currentTimeMs + 14000), + new FileAccessEvent("/file3", currentTimeMs + 14001), + new FileAccessEvent("/file3", currentTimeMs + 14002), + new FileAccessEvent("/unknown_file", currentTimeMs + 14003), + new FileAccessEvent("/file4", currentTimeMs + 16000), + new FileAccessEvent("", currentTimeMs + 22000))); + + SearchResult fileAccessInfos = + dbTableManager.search(FileAccessInfoSearchRequest.noFilters(), + PageRequest.builder() + .addSorting(FileAccessInfoSortField.FID, Sorting.Order.ASC) + .build()); + assertEquals(Arrays.asList( + new FileAccessInfo(FILE_PATH_IDS.get("/file1"), "/file1", 4, currentTimeMs + 8000), + new FileAccessInfo(FILE_PATH_IDS.get("/file2"), "/file2", 1, currentTimeMs + 14000), + new FileAccessInfo(FILE_PATH_IDS.get("/file3"), "/file3", 3, currentTimeMs + 14002), + new FileAccessInfo(FILE_PATH_IDS.get("/file4"), "/file4", 1, currentTimeMs + 16000)), + fileAccessInfos.getItems()); } private FileInfo[] testFileInfos() { @@ -134,20 +111,4 @@ private FileInfo dummyFileInfo(String path, long fileId) { .setFileId(fileId) .build(); } - - @Getter - private static class MockWindowClosedCallback implements WindowClosedCallback { - - private final List createdTables = new ArrayList<>(); - private final List - collectedAccessCounts = new ArrayList<>(); - - @Override - public void onWindowClosed(long windowStart, long windowEnd, - Collection aggregatedAccessCounts) { - AccessCountTable table = new AccessCountTable(windowStart, windowEnd); - createdTables.add(table); - collectedAccessCounts.addAll(aggregatedAccessCounts); - } - } } diff --git a/smart-metastore/src/test/java/org/smartdata/metastore/dao/accesscount/TestAddTableOpListener.java b/smart-metastore/src/test/java/org/smartdata/metastore/dao/accesscount/TestAddTableOpListener.java deleted file mode 100644 index 4002c1ff259..00000000000 --- a/smart-metastore/src/test/java/org/smartdata/metastore/dao/accesscount/TestAddTableOpListener.java +++ /dev/null @@ -1,137 +0,0 @@ -/** - * 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.smartdata.metastore.dao.accesscount; - -import org.junit.After; -import org.junit.Assert; -import org.junit.Before; -import org.junit.Test; -import org.smartdata.metastore.model.AccessCountTable; - -import java.util.concurrent.ExecutorService; -import java.util.concurrent.Executors; - -import static org.smartdata.metastore.TestDBUtil.addAccessCountTableToDeque; - -public class TestAddTableOpListener { - - private ExecutorService executorService; - - private AccessCountTableHandler tableHandler; - - @Before - public void setup() { - executorService = Executors.newFixedThreadPool(4); - tableHandler = new NoOpAccessCountTableHandler(); - } - - @After - public void shutDown() { - executorService.shutdownNow(); - } - - @Test - public void testMinuteTableListener() throws Exception { - long oneSec = 1000L; - AccessCountTableEvictor tableEvictor = new CountTableEvictor(tableHandler, 10); - AccessCountTableDeque minuteTableDeque = new AccessCountTableDeque(tableEvictor); - TableAddOpListener minuteTableListener = - TableAddOpListener.perMinute(minuteTableDeque, tableHandler, executorService); - AccessCountTableDeque secondTableDeque = - new AccessCountTableDeque(tableEvictor, minuteTableListener); - - AccessCountTable table1 = - new AccessCountTable(45 * oneSec, 50 * oneSec); - AccessCountTable table2 = - new AccessCountTable(50 * oneSec, 55 * oneSec); - AccessCountTable table3 = - new AccessCountTable(55 * oneSec, 60 * oneSec); - - addAccessCountTableToDeque(secondTableDeque, table1); - Assert.assertTrue(minuteTableDeque.isEmpty()); - - addAccessCountTableToDeque(secondTableDeque, table2); - Assert.assertTrue(minuteTableDeque.isEmpty()); - - addAccessCountTableToDeque(secondTableDeque, table3); - Assert.assertEquals(1, minuteTableDeque.size()); - - AccessCountTable expected = new AccessCountTable(0L, 60 * oneSec); - Assert.assertEquals(minuteTableDeque.poll(), expected); - } - - @Test - public void testHourTableListener() throws Exception { - long oneMin = 60 * 1000L; - AccessCountTableEvictor tableEvictor = new CountTableEvictor(tableHandler, 10); - AccessCountTableDeque hourTableDeque = new AccessCountTableDeque(tableEvictor); - TableAddOpListener hourTableListener = - TableAddOpListener.perHour(hourTableDeque, tableHandler, executorService); - AccessCountTableDeque minuteTableDeque = - new AccessCountTableDeque(tableEvictor, hourTableListener); - - AccessCountTable table1 = - new AccessCountTable(57 * oneMin, 58 * oneMin); - AccessCountTable table2 = - new AccessCountTable(58 * oneMin, 59 * oneMin); - AccessCountTable table3 = - new AccessCountTable(59 * oneMin, 60 * oneMin); - - addAccessCountTableToDeque(minuteTableDeque, table1); - Assert.assertTrue(hourTableDeque.isEmpty()); - - addAccessCountTableToDeque(minuteTableDeque, table2); - Assert.assertTrue(hourTableDeque.isEmpty()); - - addAccessCountTableToDeque(minuteTableDeque, table3); - Assert.assertEquals(1, hourTableDeque.size()); - - AccessCountTable expected = new AccessCountTable(0L, 60 * oneMin); - Assert.assertEquals(hourTableDeque.poll(), expected); - } - - @Test - public void testDayTableListener() throws Exception { - long oneHour = 60 * 60 * 1000L; - AccessCountTableEvictor tableEvictor = new CountTableEvictor(tableHandler, 10); - AccessCountTableDeque dayTableDeque = new AccessCountTableDeque(tableEvictor); - TableAddOpListener dayTableListener = - TableAddOpListener.perDay(dayTableDeque, tableHandler, executorService); - AccessCountTableDeque hourTableDeque = - new AccessCountTableDeque(tableEvictor, dayTableListener); - - AccessCountTable table1 = - new AccessCountTable(21 * oneHour, 22 * oneHour); - AccessCountTable table2 = - new AccessCountTable(22 * oneHour, 23 * oneHour); - AccessCountTable table3 = - new AccessCountTable(23 * oneHour, 24 * oneHour); - - addAccessCountTableToDeque(hourTableDeque, table1); - Assert.assertTrue(dayTableDeque.isEmpty()); - - addAccessCountTableToDeque(hourTableDeque, table2); - Assert.assertTrue(dayTableDeque.isEmpty()); - - addAccessCountTableToDeque(hourTableDeque, table3); - Assert.assertEquals(1, dayTableDeque.size()); - - AccessCountTable today = new AccessCountTable(0L, 24 * oneHour); - Assert.assertEquals(dayTableDeque.poll(), today); - } -} diff --git a/smart-metastore/src/test/java/org/smartdata/metastore/dao/accesscount/TestDbAccessCountTableManager.java b/smart-metastore/src/test/java/org/smartdata/metastore/dao/accesscount/TestDbAccessCountTableManager.java deleted file mode 100644 index a223bd97fcf..00000000000 --- a/smart-metastore/src/test/java/org/smartdata/metastore/dao/accesscount/TestDbAccessCountTableManager.java +++ /dev/null @@ -1,127 +0,0 @@ -/** - * 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.smartdata.metastore.dao.accesscount; - -import org.junit.Before; -import org.junit.Test; -import org.smartdata.metastore.TestDaoBase; -import org.smartdata.metastore.db.metadata.DbMetadataProvider; -import org.smartdata.metastore.model.AccessCountTable; - -import java.util.ArrayList; -import java.util.Arrays; -import java.util.List; - -import static org.junit.Assert.assertEquals; -import static org.junit.Assert.assertFalse; -import static org.junit.Assert.assertTrue; - -public class TestDbAccessCountTableManager extends TestDaoBase { - - private DbAccessCountTableManager dbAccessCountTableManager; - private DbMetadataProvider dbMetadataProvider; - private AccessCountTableDao accessCountTableDao; - - @Before - public void setUp() { - dbAccessCountTableManager = new DbAccessCountTableManager(metaStore); - accessCountTableDao = metaStore.accessCountTableDao(); - dbMetadataProvider = metaStore.dbMetadataProvider(); - } - - private List createTables(String... tables) throws Exception { - List createdTables = new ArrayList<>(); - - for (String tableName : tables) { - AccessCountTable table = dummyTable(tableName); - dbAccessCountTableManager.createTable(table); - createdTables.add(table); - } - - return createdTables; - } - - private AccessCountTable dummyTable(String tableName) { - return new AccessCountTable(tableName, 0L, 0L, false); - } - - @Test - public void testCreateTable() throws Exception { - createTables("table1", "table2", "table3") - .stream() - .map(AccessCountTable::getTableName) - .forEach(this::assertTableExists); - } - - @Test - public void testAggregateTables() throws Exception { - List tablesToAggregate = - createTables("table1", "table2", "table3"); - - AccessCountTable destTable = dummyTable("dest"); - dbAccessCountTableManager.aggregate(destTable, tablesToAggregate); - - assertTableExists(destTable.getTableName()); - } - - @Test - public void testDropTable() throws Exception { - List tablesToAggregate = - createTables("table1", "table2"); - - dbAccessCountTableManager.dropTable(tablesToAggregate.get(0)); - assertFalse(accessCountTableDao.tableExists("table1")); - assertFalse(dbMetadataProvider.tableExists("table1")); - assertTableExists("table2"); - - dbAccessCountTableManager.dropTable(dummyTable("another")); - assertTableExists("table2"); - } - - @Test - public void testRecoverOnlyValidTables() throws Exception { - // table from previous SSM version without last access time column - AccessCountTable oldTable = new AccessCountTable(0, 5000); - metaStore.execute( - "CREATE TABLE " + oldTable.getTableName() - + "(fid BIGINT NOT NULL, " - + "count INTEGER NOT NULL)"); - - accessCountTableDao.insert(oldTable); - - // non-existing table - AccessCountTable deletedTable = new AccessCountTable(15000, 20000); - accessCountTableDao.insert(deletedTable); - - List validTables = Arrays.asList( - new AccessCountTable(5000, 10000), - new AccessCountTable(10000, 15000), - new AccessCountTable(20000, 25000)); - for (AccessCountTable table: validTables) { - dbAccessCountTableManager.createTable(table); - } - - List validatedTables = dbAccessCountTableManager.getTables(); - assertEquals(validTables, validatedTables); - } - - private void assertTableExists(String tableName) { - assertTrue(accessCountTableDao.tableExists(tableName)); - assertTrue(dbMetadataProvider.tableExists(tableName)); - } -} diff --git a/smart-metastore/src/test/java/org/smartdata/metastore/dao/accesscount/TestFileAccessManager.java b/smart-metastore/src/test/java/org/smartdata/metastore/dao/accesscount/TestFileAccessManager.java new file mode 100644 index 00000000000..cb589fb8785 --- /dev/null +++ b/smart-metastore/src/test/java/org/smartdata/metastore/dao/accesscount/TestFileAccessManager.java @@ -0,0 +1,161 @@ +/** + * 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.smartdata.metastore.dao.accesscount; + +import org.junit.Before; +import org.junit.Test; +import org.smartdata.metastore.MetaStoreException; +import org.smartdata.metastore.accesscount.FileAccessManager; +import org.smartdata.metastore.dao.Searchable; +import org.smartdata.metastore.dao.TestSearchableDao; +import org.smartdata.metastore.model.AggregatedAccessCounts; +import org.smartdata.metastore.model.SearchResult; +import org.smartdata.metastore.partition.FileAccessPartitionManager; +import org.smartdata.metastore.partition.FileAccessPartitionManagerImpl; +import org.smartdata.metastore.queries.PageRequest; +import org.smartdata.metastore.queries.sort.FileAccessInfoSortField; +import org.smartdata.metastore.queries.sort.Sorting; +import org.smartdata.metastore.transaction.TransactionRunner; +import org.smartdata.model.FileAccessInfo; +import org.smartdata.model.FileInfo; +import org.smartdata.model.TimeInterval; +import org.smartdata.model.request.FileAccessInfoSearchRequest; + +import java.time.Instant; +import java.util.Arrays; +import java.util.Collection; +import java.util.List; +import java.util.stream.IntStream; + +import static org.junit.Assert.assertEquals; + +public class TestFileAccessManager extends + TestSearchableDao { + + private static final List TEST_FILES = Arrays.asList( + "/file0", + "/file1", + "/file2", + "/file3", + "/file4", + "/file5" + ); + private FileAccessManager fileAccessManager; + + @Before + public void setUp() { + FileAccessPartitionManager fileAccessPartitionManager = + new FileAccessPartitionManagerImpl(metaStore); + fileAccessPartitionManager.createNewPartitions(); + fileAccessManager = new FileAccessManager( + new TransactionRunner(metaStore.transactionManager()), + metaStore.accessCountEventDao(), + metaStore.cacheFileDao()); + } + + @Override + protected Searchable searchable() { + return fileAccessManager; + } + + @Override + protected Long getIdentifier(FileAccessInfo fileAccessInfo) { + return fileAccessInfo.getFid(); + } + + @Override + protected FileAccessInfoSortField defaultSortField() { + return FileAccessInfoSortField.FID; + } + + @Test + public void testSaveAccessCounts() throws MetaStoreException { + long currentTimeMillis = System.currentTimeMillis(); + insertFileAccessCounts(currentTimeMillis); + + SearchResult fileAccessInfos = + fileAccessManager.search(FileAccessInfoSearchRequest.noFilters(), + PageRequest.builder() + .addSorting(FileAccessInfoSortField.FID, Sorting.Order.ASC) + .build()); + assertEquals(Arrays.asList(new FileAccessInfo(1, TEST_FILES.get(1), 3, currentTimeMillis + 2), + new FileAccessInfo(2, TEST_FILES.get(2), 1, currentTimeMillis + 2), + new FileAccessInfo(3, TEST_FILES.get(3), 2, currentTimeMillis + 3)), + fileAccessInfos.getItems()); + } + + @Test + public void testSearchByIds() throws MetaStoreException { + List ids = Arrays.asList(1L, 2L); + long currentTimeMillis = System.currentTimeMillis(); + insertFileAccessCounts(currentTimeMillis); + testSearch(FileAccessInfoSearchRequest.builder().ids(ids).build(), 1L, 2L); + } + + @Test + public void testSearchByPath() throws MetaStoreException { + long currentTimeMillis = System.currentTimeMillis(); + insertFileAccessCounts(currentTimeMillis); + testSearch(FileAccessInfoSearchRequest.builder() + .pathLike("/file3") + .build(), + 3L); + testSearch(FileAccessInfoSearchRequest.builder() + .pathLike("/file") + .build(), + 1L, 2L, 3L); + } + + @Test + public void testSearchByLastAccessTime() throws MetaStoreException { + long currentTimeMillis = System.currentTimeMillis(); + insertFileAccessCounts(currentTimeMillis); + + testSearch(FileAccessInfoSearchRequest.builder() + .lastAccessedTime(TimeInterval.builder() + .from(Instant.ofEpochMilli(currentTimeMillis)) + .to(Instant.ofEpochMilli(currentTimeMillis + 2)) + .build()) + .build(), + 1L, 2L); + } + + private void insertFileAccessCounts(long currentTimeMillis) throws MetaStoreException { + createTestFiles(); + Collection accessCounts = Arrays.asList( + new AggregatedAccessCounts(1, 1, currentTimeMillis), + new AggregatedAccessCounts(1, 1, currentTimeMillis + 1), + new AggregatedAccessCounts(1, 1, currentTimeMillis + 2), + new AggregatedAccessCounts(2, 1, currentTimeMillis + 2), + new AggregatedAccessCounts(3, 1, currentTimeMillis + 2), + new AggregatedAccessCounts(3, 1, currentTimeMillis + 3) + ); + metaStore.accessCountEventDao().insert(accessCounts); + } + + private void createTestFiles() throws MetaStoreException { + FileInfo[] fileInfos = IntStream.range(0, TEST_FILES.size()) + .mapToObj(id -> FileInfo.newBuilder() + .setFileId(id) + .setPath(TEST_FILES.get(id)) + .build()) + .toArray(FileInfo[]::new); + + metaStore.insertFiles(fileInfos); + } +} diff --git a/smart-metastore/src/test/java/org/smartdata/metastore/partition/FileAccessPartitionManagerImplTest.java b/smart-metastore/src/test/java/org/smartdata/metastore/partition/FileAccessPartitionManagerImplTest.java new file mode 100644 index 00000000000..eef65b21329 --- /dev/null +++ b/smart-metastore/src/test/java/org/smartdata/metastore/partition/FileAccessPartitionManagerImplTest.java @@ -0,0 +1,57 @@ +/** + * 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.smartdata.metastore.partition; + +import org.junit.Before; +import org.junit.Test; +import org.smartdata.metastore.TestDaoBase; +import org.smartdata.metastore.model.FileAccessPartition; + +import java.time.LocalDate; +import java.time.format.DateTimeFormatter; +import java.util.Arrays; +import java.util.List; +import java.util.stream.Collectors; + +import static org.junit.Assert.assertEquals; + +public class FileAccessPartitionManagerImplTest extends TestDaoBase { + private static final String FILE_ACCESS_PARTITION_NAME_TEMPLATE = "file_access_%s"; + private FileAccessPartitionManager fileAccessPartitionManager; + private static final DateTimeFormatter PARTITION_DATE_TIME_FORMAT = + DateTimeFormatter.ofPattern("yyyy_MM_dd"); + + @Before + public void setUp() throws Exception { + fileAccessPartitionManager = new FileAccessPartitionManagerImpl(metaStore); + } + + @Test + public void testCreatePartitions() { + LocalDate currentDate = LocalDate.now().withDayOfMonth(1); + String currentMonthPartition = + String.format(FILE_ACCESS_PARTITION_NAME_TEMPLATE, + currentDate.format(PARTITION_DATE_TIME_FORMAT)); + String nextMonthPartition = String.format(FILE_ACCESS_PARTITION_NAME_TEMPLATE, + currentDate.plusMonths(1).withDayOfMonth(1).format(PARTITION_DATE_TIME_FORMAT)); + fileAccessPartitionManager.createNewPartitions(); + List partitions = metaStore.fileAccessPartitionDao().getAll(); + assertEquals(Arrays.asList(currentMonthPartition, nextMonthPartition), partitions.stream().map( + FileAccessPartition::getName).collect(Collectors.toList())); + } +} diff --git a/smart-metastore/src/test/java/org/smartdata/metastore/partition/cleanup/impl/MonthCountFileAccessPartitionRetentionPolicyExecutorTest.java b/smart-metastore/src/test/java/org/smartdata/metastore/partition/cleanup/impl/MonthCountFileAccessPartitionRetentionPolicyExecutorTest.java new file mode 100644 index 00000000000..3dd4a21adfa --- /dev/null +++ b/smart-metastore/src/test/java/org/smartdata/metastore/partition/cleanup/impl/MonthCountFileAccessPartitionRetentionPolicyExecutorTest.java @@ -0,0 +1,104 @@ +/** + * 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.smartdata.metastore.partition.cleanup.impl; + +import org.junit.Test; +import org.smartdata.metastore.dao.FileAccessPartitionDao; +import org.smartdata.metastore.model.FileAccessPartition; +import org.smartdata.metastore.partition.cleanup.FileAccessPartitionRetentionPolicyExecutor; + +import java.time.LocalDate; +import java.util.Arrays; +import java.util.List; + +import static org.mockito.Matchers.any; +import static org.mockito.Matchers.eq; +import static org.mockito.Mockito.mock; +import static org.mockito.Mockito.never; +import static org.mockito.Mockito.times; +import static org.mockito.Mockito.verify; +import static org.mockito.Mockito.when; + +public class MonthCountFileAccessPartitionRetentionPolicyExecutorTest { + + private FileAccessPartitionRetentionPolicyExecutor retentionPolicyExecutor; + private final FileAccessPartitionDao mockFileAccessPartitionDao = + mock(FileAccessPartitionDao.class); + + @Test + public void cleanupRedundantPartitions() { + int retentionCount = 2; + retentionPolicyExecutor = + new MonthCountFileAccessPartitionRetentionPolicyExecutor(mockFileAccessPartitionDao, + retentionCount); + List partitions = Arrays.asList( + new FileAccessPartition(1, "2024-07-01", + LocalDate.of(2024, 7, 1)), + new FileAccessPartition(2, "2024-08-01", + LocalDate.of(2024, 8, 1)), + new FileAccessPartition(3, "2024-09-01", + LocalDate.of(2024, 9, 1)), + new FileAccessPartition(4, "2024-10-01", + LocalDate.of(2024, 10, 1)), + new FileAccessPartition(5, "2024-11-01", + LocalDate.of(2024, 11, 1)) + ); + when(mockFileAccessPartitionDao.getAll()).thenReturn(partitions); + retentionPolicyExecutor.cleanup(); + verify(mockFileAccessPartitionDao, times(1)).remove(eq(partitions.get(3))); + verify(mockFileAccessPartitionDao, times(1)).remove(eq(partitions.get(4))); + } + + @Test + public void testCleanupWithoutRedundantPartitions() { + int retentionCount = 2; + retentionPolicyExecutor = + new MonthCountFileAccessPartitionRetentionPolicyExecutor(mockFileAccessPartitionDao, + retentionCount); + List partitions = Arrays.asList( + new FileAccessPartition(1, "2024-07-01", + LocalDate.of(2024, 7, 1)), + new FileAccessPartition(2, "2024-08-01", + LocalDate.of(2024, 8, 1)), + new FileAccessPartition(3, "2024-09-01", + LocalDate.of(2024, 9, 1)) + ); + when(mockFileAccessPartitionDao.getAll()).thenReturn(partitions); + retentionPolicyExecutor.cleanup(); + verify(mockFileAccessPartitionDao, never()).remove(any()); + } + + @Test + public void testCleanupWithIncorrectRetentionCount() { + int retentionCount = -1; + retentionPolicyExecutor = + new MonthCountFileAccessPartitionRetentionPolicyExecutor(mockFileAccessPartitionDao, + retentionCount); + List partitions = Arrays.asList( + new FileAccessPartition(1, "2024-07-01", + LocalDate.of(2024, 7, 1)), + new FileAccessPartition(2, "2024-08-01", + LocalDate.of(2024, 8, 1)), + new FileAccessPartition(3, "2024-09-01", + LocalDate.of(2024, 9, 1)) + ); + retentionPolicyExecutor.cleanup(); + verify(mockFileAccessPartitionDao, never()).getAll(); + verify(mockFileAccessPartitionDao, never()).remove(any()); + } +} diff --git a/smart-metastore/src/test/resources/druid-template.xml b/smart-metastore/src/test/resources/druid-template.xml index daa1dc33d8a..dee3700feee 100644 --- a/smart-metastore/src/test/resources/druid-template.xml +++ b/smart-metastore/src/test/resources/druid-template.xml @@ -1,14 +1,12 @@ - sqlite - jdbc:sqlite:sqlit-db-path "" "" - 1 - 1 - 2 + 8 + 4 + 16 60000 90000 diff --git a/smart-rule/src/main/java/org/smartdata/rule/objects/FileObject.java b/smart-rule/src/main/java/org/smartdata/rule/objects/FileObject.java index e880035bfd5..6713e859310 100644 --- a/smart-rule/src/main/java/org/smartdata/rule/objects/FileObject.java +++ b/smart-rule/src/main/java/org/smartdata/rule/objects/FileObject.java @@ -59,18 +59,10 @@ public class FileObject extends SmartObject { new Property("acBot", ValueType.LONG, Arrays.asList(ValueType.TIMEINTVAL, ValueType.LONG), "VIRTUAL_ACCESS_COUNT_TABLE", "", false, "count")); - PROPERTIES.put("accessCountTopOnStoragePolicy", - new Property("accessCountTopOnStoragePolicy", ValueType.LONG, - Arrays.asList(ValueType.TIMEINTVAL, ValueType.LONG, ValueType.STRING), - "VIRTUAL_ACCESS_COUNT_TABLE", "", false, "count")); PROPERTIES.put("acTopSp", new Property("acTopSp", ValueType.LONG, Arrays.asList(ValueType.TIMEINTVAL, ValueType.LONG, ValueType.STRING), "VIRTUAL_ACCESS_COUNT_TABLE", "", false, "count")); - PROPERTIES.put("accessCountBottomOnStoragePolicy", - new Property("accessCountBottomOnStoragePolicy", ValueType.LONG, - Arrays.asList(ValueType.TIMEINTVAL, ValueType.LONG, ValueType.STRING), - "VIRTUAL_ACCESS_COUNT_TABLE", "", false, "count")); PROPERTIES.put("acBotSp", new Property("acBotSp", ValueType.LONG, Arrays.asList(ValueType.TIMEINTVAL, ValueType.LONG, ValueType.STRING), diff --git a/smart-rule/src/main/java/org/smartdata/rule/parser/SmartRuleVisitTranslator.java b/smart-rule/src/main/java/org/smartdata/rule/parser/SmartRuleVisitTranslator.java index 27c3fe6abb5..af45e2522ff 100644 --- a/smart-rule/src/main/java/org/smartdata/rule/parser/SmartRuleVisitTranslator.java +++ b/smart-rule/src/main/java/org/smartdata/rule/parser/SmartRuleVisitTranslator.java @@ -914,28 +914,6 @@ public NodeTransResult doGenerateSql(TreeNode root, String tableName) throws IOE return new NodeTransResult(null, "$" + mStrValue); } - if (p.getPropertyName().equals("accessCountTopOnStoragePolicy") - || p.getPropertyName().equals("accessCountBottomOnStoragePolicy") - || p.getPropertyName().equals("acTopSp") - || p.getPropertyName().equals("acBotSp")) { - boolean topFlag = p.getPropertyName().equals("accessCountTopOnStoragePolicy") - || p.getPropertyName().equals("acTopSp"); - String virTab = genAccessCountTable(transCtx == null ? 0 : transCtx.getRuleId(), - (Long) realParas.getValues().get(0)); - String func = "$@genVirtualAccessCountTable" + (topFlag ? "Top" : "Bottom") - + "ValueOnStoragePolicy"; - String mStr = virTab + (topFlag ? "_top_" : "_bottom_") - + realParas.getValues().get(1).toString() + "_on_storage_policy_" - + realParas.getValues().get(2).toString(); - String mStrValue = mStr + "_value"; - if (!sqlStatements.contains(func + "(" + mStr + ")")) { - sqlStatements.add(func + "(" + mStr + ")"); - dynamicParameters.put(mStr, Arrays.asList(realParas.getValues(), virTab, mStrValue)); - } - procAcc = true; - return new NodeTransResult(null, "$" + mStrValue); - } - return new NodeTransResult(p.getTableName(), realParas.formatParameters()); } } diff --git a/smart-rule/src/test/java/org/smartdata/rule/TestSmartRuleStringParser.java b/smart-rule/src/test/java/org/smartdata/rule/TestSmartRuleStringParser.java index 15e0885ce00..5aba7d066c1 100644 --- a/smart-rule/src/test/java/org/smartdata/rule/TestSmartRuleStringParser.java +++ b/smart-rule/src/test/java/org/smartdata/rule/TestSmartRuleStringParser.java @@ -36,11 +36,7 @@ public void testRuleTranslate() throws Exception { rules.add("file : ac(10min) > acTop(10min, 10) | sleep -ms 0"); rules.add("file : accessCount(10min) > accessCountBottom(10min, 10) | sleep -ms 0"); rules.add("file : ac(10min) > acBot(10min, 10) | sleep -ms 0"); - rules.add("file : ac(10min) > accessCountTopOnStoragePolicy(10min, 10, \"ALL_SSD\") " - + "| sleep -ms 0"); rules.add("file : ac(10min) > acTopSp(10min, 10, \"ALL_SSD\") | sleep -ms 0"); - rules.add("file : ac(10min) > accessCountBottomOnStoragePolicy(10min, 10, \"CACHE\") " - + "| sleep -ms 0"); rules.add("file : ac(10min) > acBotSp(10min, 10, \"CACHE\") | sleep -ms 0"); rules.add("file : ac(10min) > acBotSp(10min, 10, \"HOT\") and acBotSp(10min, 10, \"HOT\") > 0 " + "| sleep -ms 0"); diff --git a/smart-server/src/test/java/org/smartdata/server/MiniSmartClusterHarness.java b/smart-server/src/test/java/org/smartdata/server/MiniSmartClusterHarness.java index c9a83975844..6a1c4d0edf3 100644 --- a/smart-server/src/test/java/org/smartdata/server/MiniSmartClusterHarness.java +++ b/smart-server/src/test/java/org/smartdata/server/MiniSmartClusterHarness.java @@ -25,8 +25,6 @@ import org.smartdata.conf.SmartConf; import org.smartdata.conf.SmartConfKeys; import org.smartdata.hdfs.MiniClusterWithStoragesHarness; -import org.smartdata.metastore.TestDBUtil; -import org.smartdata.metastore.utils.MetaStoreUtils; import java.io.IOException; import java.net.URI; @@ -49,11 +47,7 @@ public void init() throws Exception { List uriList = new ArrayList<>(namenodes); conf.set(DFS_NAMENODE_HTTP_ADDRESS_KEY, uriList.get(0).toString()); conf.set(SmartConfKeys.SMART_DFS_NAMENODE_RPCSERVER_KEY, - uriList.get(0).toString()); - - String dbFile = TestDBUtil.getUniqueEmptySqliteDBFile(); - String dbUrl = MetaStoreUtils.SQLITE_URL_PREFIX + dbFile; - smartContext.getConf().set(SmartConfKeys.SMART_METASTORE_DB_URL_KEY, dbUrl); + uriList.get(0).toString()); // rpcServer start in SmartServer ssm = SmartServer.launchWith(conf); diff --git a/smart-server/src/test/java/org/smartdata/server/TestSmartServer.java b/smart-server/src/test/java/org/smartdata/server/TestSmartServer.java index 107793580db..a0edebe4218 100644 --- a/smart-server/src/test/java/org/smartdata/server/TestSmartServer.java +++ b/smart-server/src/test/java/org/smartdata/server/TestSmartServer.java @@ -23,15 +23,10 @@ import org.junit.Before; import org.junit.Test; import org.smartdata.conf.SmartConf; -import org.smartdata.conf.SmartConfKeys; -import org.smartdata.metastore.TestDBUtil; -import org.smartdata.metastore.utils.MetaStoreUtils; public class TestSmartServer { protected SmartConf conf; protected SmartServer ssm; - protected String dbFile; - protected String dbUrl; private static final int DEFAULT_BLOCK_SIZE = 100; @@ -44,11 +39,6 @@ public void setUp() throws Exception { conf = new SmartConf(); initConf(conf); - // Set db used - dbFile = TestDBUtil.getUniqueEmptySqliteDBFile(); - dbUrl = MetaStoreUtils.SQLITE_URL_PREFIX + dbFile; - conf.set(SmartConfKeys.SMART_METASTORE_DB_URL_KEY, dbUrl); - // rpcServer start in SmartServer ssm = SmartServer.launchWith(conf); } diff --git a/smart-server/src/test/java/org/smartdata/server/TestSmartServerCli.java b/smart-server/src/test/java/org/smartdata/server/TestSmartServerCli.java index 2edb87ef074..8bcb7f7b6f6 100644 --- a/smart-server/src/test/java/org/smartdata/server/TestSmartServerCli.java +++ b/smart-server/src/test/java/org/smartdata/server/TestSmartServerCli.java @@ -23,8 +23,6 @@ import org.smartdata.conf.SmartConf; import org.smartdata.conf.SmartConfKeys; import org.smartdata.hdfs.MiniClusterHarness; -import org.smartdata.metastore.TestDBUtil; -import org.smartdata.metastore.utils.MetaStoreUtils; import java.net.URI; import java.util.ArrayList; @@ -38,12 +36,7 @@ public void testConfNameNodeRPCAddr() throws Exception { try { Collection namenodes = DFSUtil.getInternalNsRpcUris(smartContext.getConf()); List uriList = new ArrayList<>(namenodes); - SmartConf conf = new SmartConf(); - // Set db used - String dbFile = TestDBUtil.getUniqueEmptySqliteDBFile(); - String dbUrl = MetaStoreUtils.SQLITE_URL_PREFIX + dbFile; - conf.set(SmartConfKeys.SMART_METASTORE_DB_URL_KEY, dbUrl); // rpcServer start in SmartServer SmartServer ssm = null; @@ -71,7 +64,7 @@ public void testConfNameNodeRPCAddr() throws Exception { Thread.sleep(1000); regServer.shutdown(); - args = new String[] { + args = new String[]{ "-h" }; SmartServer.launchWith(args, conf); diff --git a/smart-server/src/test/java/org/smartdata/server/TestSmartServerLogin.java b/smart-server/src/test/java/org/smartdata/server/TestSmartServerLogin.java index f1e2db82606..ed0eec1c9ed 100644 --- a/smart-server/src/test/java/org/smartdata/server/TestSmartServerLogin.java +++ b/smart-server/src/test/java/org/smartdata/server/TestSmartServerLogin.java @@ -27,8 +27,6 @@ import org.smartdata.conf.SmartConf; import org.smartdata.conf.SmartConfKeys; import org.smartdata.hdfs.MiniClusterFactory; -import org.smartdata.metastore.TestDBUtil; -import org.smartdata.metastore.utils.MetaStoreUtils; import java.io.File; import java.net.URI; @@ -47,8 +45,6 @@ public class TestSmartServerLogin { private int serverPort = -1; private SmartConf conf; private MiniDFSCluster cluster; - private String dbFile; - private String dbUrl; private SmartServer ssm; private final String keytabFileName = "smart.keytab"; @@ -74,12 +70,6 @@ private void initConf() throws Exception { conf.set(DFS_NAMENODE_HTTP_ADDRESS_KEY, uriList.get(0).toString()); conf.set(SmartConfKeys.SMART_DFS_NAMENODE_RPCSERVER_KEY, uriList.get(0).toString()); - - // Set db used - dbFile = TestDBUtil.getUniqueEmptySqliteDBFile(); - dbUrl = MetaStoreUtils.SQLITE_URL_PREFIX + dbFile; - conf.set(SmartConfKeys.SMART_METASTORE_DB_URL_KEY, dbUrl); - conf.setBoolean(SmartConfKeys.SMART_SECURITY_ENABLE, true); conf.set(SmartConfKeys.SMART_SERVER_KEYTAB_FILE_KEY, keytabFileName); conf.set(SmartConfKeys.SMART_SERVER_KERBEROS_PRINCIPAL_KEY, principal); diff --git a/smart-server/src/test/java/org/smartdata/server/engine/audit/TestCmdletLifecycleLogger.java b/smart-server/src/test/java/org/smartdata/server/engine/audit/TestCmdletLifecycleLogger.java index 862de859842..13e6759d6d6 100644 --- a/smart-server/src/test/java/org/smartdata/server/engine/audit/TestCmdletLifecycleLogger.java +++ b/smart-server/src/test/java/org/smartdata/server/engine/audit/TestCmdletLifecycleLogger.java @@ -22,7 +22,7 @@ import org.junit.Before; import org.junit.Test; import org.smartdata.conf.SmartConf; -import org.smartdata.metastore.SqliteTestDaoBase; +import org.smartdata.metastore.TestDaoBase; import org.smartdata.model.CmdletInfo; import org.smartdata.model.audit.UserActivityEvent; import org.smartdata.model.request.AuditSearchRequest; @@ -46,7 +46,7 @@ import static org.smartdata.model.audit.UserActivityResult.FAILURE; import static org.smartdata.model.audit.UserActivityResult.SUCCESS; -public class TestCmdletLifecycleLogger extends SqliteTestDaoBase { +public class TestCmdletLifecycleLogger extends TestDaoBase { private CmdletManager cmdletManager; private AuditService auditService; private SmartPrincipalManager principalManager; diff --git a/smart-server/src/test/java/org/smartdata/server/engine/audit/TestRuleLifecycleLogger.java b/smart-server/src/test/java/org/smartdata/server/engine/audit/TestRuleLifecycleLogger.java index edd75ddfb06..8583c98f8d6 100644 --- a/smart-server/src/test/java/org/smartdata/server/engine/audit/TestRuleLifecycleLogger.java +++ b/smart-server/src/test/java/org/smartdata/server/engine/audit/TestRuleLifecycleLogger.java @@ -22,7 +22,7 @@ import org.junit.Before; import org.junit.Test; import org.smartdata.conf.SmartConf; -import org.smartdata.metastore.SqliteTestDaoBase; +import org.smartdata.metastore.TestDaoBase; import org.smartdata.model.RuleInfo; import org.smartdata.model.RuleState; import org.smartdata.model.audit.UserActivityEvent; @@ -47,7 +47,7 @@ import static org.smartdata.model.audit.UserActivityResult.FAILURE; import static org.smartdata.model.audit.UserActivityResult.SUCCESS; -public class TestRuleLifecycleLogger extends SqliteTestDaoBase { +public class TestRuleLifecycleLogger extends TestDaoBase { private RuleManager ruleManager; private AuditService auditService; private SmartPrincipalManager principalManager; diff --git a/smart-server/src/test/resources/druid-template.xml b/smart-server/src/test/resources/druid-template.xml index 7e2e89d5ccc..278346ba9d6 100644 --- a/smart-server/src/test/resources/druid-template.xml +++ b/smart-server/src/test/resources/druid-template.xml @@ -1,14 +1,14 @@ - sqlite - jdbc:sqlite:sqlit-db-path + jdbc:tc:postgresql:12.19:///ssm_postgres + org.testcontainers.jdbc.ContainerDatabaseDriver "" "" - 1 - 1 - 1 + 8 + 4 + 16 60000 90000 diff --git a/smart-web-server/src/main/java/org/smartdata/server/config/SsmContextInitializer.java b/smart-web-server/src/main/java/org/smartdata/server/config/SsmContextInitializer.java index 0dcf353eee2..6f0fd80fae1 100644 --- a/smart-web-server/src/main/java/org/smartdata/server/config/SsmContextInitializer.java +++ b/smart-web-server/src/main/java/org/smartdata/server/config/SsmContextInitializer.java @@ -51,7 +51,7 @@ public void initialize(ConfigurableApplicationContext applicationContext) { "cachedFilesManager", smartEngine.getStatesManager().getCachedFilesManager()); beanFactory.registerSingleton( "smartPrincipalManager", smartEngine.getSmartPrincipalManager()); - beanFactory.registerSingleton("accessCountTableManager", - smartEngine.getStatesManager().getAccessCountTableManager()); + beanFactory.registerSingleton("dbFileAccessManager", + smartEngine.getStatesManager().getFileAccessManager()); } } diff --git a/smart-web-server/src/main/java/org/smartdata/server/controller/FilesControllerDelegate.java b/smart-web-server/src/main/java/org/smartdata/server/controller/FilesControllerDelegate.java index d9371e25d88..b52affc6276 100644 --- a/smart-web-server/src/main/java/org/smartdata/server/controller/FilesControllerDelegate.java +++ b/smart-web-server/src/main/java/org/smartdata/server/controller/FilesControllerDelegate.java @@ -18,7 +18,7 @@ package org.smartdata.server.controller; import lombok.RequiredArgsConstructor; -import org.smartdata.metastore.dao.accesscount.AccessCountTableManager; +import org.smartdata.metastore.accesscount.FileAccessManager; import org.smartdata.metastore.model.SearchResult; import org.smartdata.metastore.queries.PageRequest; import org.smartdata.metastore.queries.sort.FileAccessInfoSortField; @@ -44,7 +44,7 @@ @RequiredArgsConstructor public class FilesControllerDelegate implements FilesApiDelegate { - private final AccessCountTableManager accessCountTableManager; + private final FileAccessManager fileAccessManager; private final FileAccessInfoMapper fileInfoMapper; private final FileAccessInfoPageRequestMapper pageRequestMapper; private final CachedFilesControllerDelegate cachedFilesControllerDelegate; @@ -72,7 +72,7 @@ public FileAccessCountsDto getAccessCounts(PageRequestDto pageRequestDto, lastAccessedTime); SearchResult searchResult = - accessCountTableManager.search(searchRequest, pageRequest); + fileAccessManager.search(searchRequest, pageRequest); return fileInfoMapper.toFileAccessCountsDto(searchResult); } } diff --git a/smart-web-server/src/main/java/org/smartdata/server/mappers/FileAccessInfoMapper.java b/smart-web-server/src/main/java/org/smartdata/server/mappers/FileAccessInfoMapper.java index bf9b58d6594..299beb11a2e 100644 --- a/smart-web-server/src/main/java/org/smartdata/server/mappers/FileAccessInfoMapper.java +++ b/smart-web-server/src/main/java/org/smartdata/server/mappers/FileAccessInfoMapper.java @@ -37,7 +37,6 @@ public interface FileAccessInfoMapper extends SmartMapper { @Mapping(target = "ids", ignore = true) @Mapping(target = "id", ignore = true) - @Mapping(target = "accessCountTables", ignore = true) FileAccessInfoSearchRequest toSearchRequest(String pathLike, LastAccessedTimeIntervalDto lastAccessedTime); diff --git a/smart-zeppelin/zeppelin-web/src/app/dashboard/views/rules/submit/help.html b/smart-zeppelin/zeppelin-web/src/app/dashboard/views/rules/submit/help.html new file mode 100755 index 00000000000..e69de29bb2d