All Downloads are FREE. Search and download functionalities are using the official Maven repository.

io.shardingsphere.core.rule.ShardingRule Maven / Gradle / Ivy

There is a newer version: 3.1.0
Show newest version
/*
 * Copyright 2016-2018 shardingsphere.io.
 * 

* Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. * You may obtain a copy of the License at * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. * See the License for the specific language governing permissions and * limitations under the License. *

*/ package io.shardingsphere.core.rule; import com.google.common.base.Optional; import com.google.common.base.Preconditions; import com.google.common.base.Strings; import io.shardingsphere.core.api.config.MasterSlaveRuleConfiguration; import io.shardingsphere.core.api.config.ShardingRuleConfiguration; import io.shardingsphere.core.api.config.TableRuleConfiguration; import io.shardingsphere.core.exception.ShardingConfigurationException; import io.shardingsphere.core.exception.ShardingException; import io.shardingsphere.core.keygen.DefaultKeyGenerator; import io.shardingsphere.core.keygen.KeyGenerator; import io.shardingsphere.core.parsing.parser.context.condition.Column; import io.shardingsphere.core.routing.strategy.ShardingStrategy; import io.shardingsphere.core.routing.strategy.ShardingStrategyFactory; import io.shardingsphere.core.routing.strategy.none.NoneShardingStrategy; import io.shardingsphere.core.util.StringUtil; import lombok.Getter; import java.util.Collection; import java.util.Collections; import java.util.LinkedList; import java.util.List; import java.util.TreeSet; /** * Databases and tables sharding rule configuration. * * @author zhangliang * @author maxiaoguang * @author panjuan */ @Getter public final class ShardingRule { private final ShardingRuleConfiguration shardingRuleConfig; private final ShardingDataSourceNames shardingDataSourceNames; private final Collection tableRules = new LinkedList<>(); private final Collection bindingTableRules = new LinkedList<>(); private final ShardingStrategy defaultDatabaseShardingStrategy; private final ShardingStrategy defaultTableShardingStrategy; private final KeyGenerator defaultKeyGenerator; private final Collection masterSlaveRules = new LinkedList<>(); public ShardingRule(final ShardingRuleConfiguration shardingRuleConfig, final Collection dataSourceNames) { Preconditions.checkNotNull(dataSourceNames, "Data sources cannot be null."); Preconditions.checkArgument(!dataSourceNames.isEmpty(), "Data sources cannot be empty."); this.shardingRuleConfig = shardingRuleConfig; shardingDataSourceNames = new ShardingDataSourceNames(shardingRuleConfig, dataSourceNames); for (TableRuleConfiguration each : shardingRuleConfig.getTableRuleConfigs()) { tableRules.add(new TableRule(each, shardingDataSourceNames)); } for (String group : shardingRuleConfig.getBindingTableGroups()) { List tableRulesForBinding = new LinkedList<>(); for (String logicTableNameForBindingTable : StringUtil.splitWithComma(group)) { tableRulesForBinding.add(getTableRuleByLogicTableName(logicTableNameForBindingTable)); } bindingTableRules.add(new BindingTableRule(tableRulesForBinding)); } defaultDatabaseShardingStrategy = null == shardingRuleConfig.getDefaultDatabaseShardingStrategyConfig() ? new NoneShardingStrategy() : ShardingStrategyFactory.newInstance(shardingRuleConfig.getDefaultDatabaseShardingStrategyConfig()); defaultTableShardingStrategy = null == shardingRuleConfig.getDefaultTableShardingStrategyConfig() ? new NoneShardingStrategy() : ShardingStrategyFactory.newInstance(shardingRuleConfig.getDefaultTableShardingStrategyConfig()); defaultKeyGenerator = null == shardingRuleConfig.getDefaultKeyGenerator() ? new DefaultKeyGenerator() : shardingRuleConfig.getDefaultKeyGenerator(); for (MasterSlaveRuleConfiguration each : shardingRuleConfig.getMasterSlaveRuleConfigs()) { masterSlaveRules.add(new MasterSlaveRule(each)); } } /** * Try to find table rule though logic table name. * * @param logicTableName logic table name * @return table rule */ public Optional tryFindTableRuleByLogicTable(final String logicTableName) { for (TableRule each : tableRules) { if (each.getLogicTable().equals(logicTableName.toLowerCase())) { return Optional.of(each); } } return Optional.absent(); } /** * Try to find table rule though actual table name. * * @param actualTableName actual table name * @return table rule */ public Optional tryFindTableRuleByActualTable(final String actualTableName) { for (TableRule each : tableRules) { if (each.isExisted(actualTableName)) { return Optional.of(each); } } return Optional.absent(); } /** * Find table rule though actual table name. * * @param actualTableName actual table name * @return table rule */ public TableRule getTableRuleByActualTableName(final String actualTableName) { Optional tableRule = tryFindTableRuleByActualTable(actualTableName.toLowerCase()); if (tableRule.isPresent()) { return tableRule.get(); } if (!Strings.isNullOrEmpty(shardingDataSourceNames.getDefaultDataSourceName())) { return createTableRuleWithDefaultDataSource(actualTableName.toLowerCase()); } throw new ShardingConfigurationException("Cannot find table rule and default data source with actual table: '%s'", actualTableName); } /** * Find table rule though logic table name. * * @param logicTableName logic table name * @return table rule */ public TableRule getTableRuleByLogicTableName(final String logicTableName) { Optional tableRule = tryFindTableRuleByLogicTable(logicTableName.toLowerCase()); if (tableRule.isPresent()) { return tableRule.get(); } if (!Strings.isNullOrEmpty(shardingDataSourceNames.getDefaultDataSourceName())) { return createTableRuleWithDefaultDataSource(logicTableName.toLowerCase()); } throw new ShardingConfigurationException("Cannot find table rule and default data source with logic table: '%s'", logicTableName); } private TableRule createTableRuleWithDefaultDataSource(final String logicTableName) { TableRuleConfiguration tableRuleConfig = new TableRuleConfiguration(); tableRuleConfig.setLogicTable(logicTableName); tableRuleConfig.setActualDataNodes(shardingDataSourceNames.getDefaultDataSourceName() + "." + logicTableName); return new TableRule(tableRuleConfig, new ShardingDataSourceNames(shardingRuleConfig, Collections.singletonList(shardingDataSourceNames.getDefaultDataSourceName()))); } /** * Get database sharding strategy. * *

* Use default database sharding strategy if not found. *

* * @param tableRule table rule * @return database sharding strategy */ public ShardingStrategy getDatabaseShardingStrategy(final TableRule tableRule) { return null == tableRule.getDatabaseShardingStrategy() ? defaultDatabaseShardingStrategy : tableRule.getDatabaseShardingStrategy(); } /** * Get table sharding strategy. * *

* Use default table sharding strategy if not found. *

* * @param tableRule table rule * @return table sharding strategy */ public ShardingStrategy getTableShardingStrategy(final TableRule tableRule) { return null == tableRule.getTableShardingStrategy() ? defaultTableShardingStrategy : tableRule.getTableShardingStrategy(); } /** * Adjust logic tables is all belong to binding tables. * * @param logicTables names of logic tables * @return logic tables is all belong to binding tables or not */ public boolean isAllBindingTables(final Collection logicTables) { if (logicTables.isEmpty()) { return false; } Optional bindingTableRule = findBindingTableRule(logicTables); if (!bindingTableRule.isPresent()) { return false; } Collection result = new TreeSet<>(String.CASE_INSENSITIVE_ORDER); result.addAll(bindingTableRule.get().getAllLogicTables()); return !result.isEmpty() && result.containsAll(logicTables); } /** * Adjust logic tables is all belong to default data source. * * @param logicTables names of logic tables * @return logic tables is all belong to default data source */ public boolean isAllInDefaultDataSource(final Collection logicTables) { for (String each : logicTables) { if (tryFindTableRuleByLogicTable(each).isPresent()) { return false; } } return !logicTables.isEmpty(); } private Optional findBindingTableRule(final Collection logicTables) { for (String each : logicTables) { Optional result = findBindingTableRule(each); if (result.isPresent()) { return result; } } return Optional.absent(); } /** * Get binding table rule via logic table name. * * @param logicTable logic table name * @return binding table rule */ public Optional findBindingTableRule(final String logicTable) { for (BindingTableRule each : bindingTableRules) { if (each.hasLogicTable(logicTable)) { return Optional.of(each); } } return Optional.absent(); } /** * Adjust is sharding column or not. * * @param column column object * @return is sharding column or not */ public boolean isShardingColumn(final Column column) { if (defaultDatabaseShardingStrategy.getShardingColumns().contains(column.getName()) || defaultTableShardingStrategy.getShardingColumns().contains(column.getName())) { return true; } for (TableRule each : tableRules) { if (!each.getLogicTable().equalsIgnoreCase(column.getTableName())) { continue; } if (null != each.getDatabaseShardingStrategy() && each.getDatabaseShardingStrategy().getShardingColumns().contains(column.getName())) { return true; } if (null != each.getTableShardingStrategy() && each.getTableShardingStrategy().getShardingColumns().contains(column.getName())) { return true; } } return false; } /** * get generated key's column. * * @param logicTableName logic table name * @return generated key's column */ public Optional getGenerateKeyColumn(final String logicTableName) { for (TableRule each : tableRules) { if (each.getLogicTable().equalsIgnoreCase(logicTableName) && null != each.getGenerateKeyColumn()) { return Optional.of(new Column(each.getGenerateKeyColumn(), logicTableName)); } } return Optional.absent(); } /** * Generate key. * * @param logicTableName logic table name * @return generated key */ public Number generateKey(final String logicTableName) { Optional tableRule = tryFindTableRuleByLogicTable(logicTableName); if (!tableRule.isPresent()) { throw new ShardingConfigurationException("Cannot find strategy for generate keys."); } if (null != tableRule.get().getKeyGenerator()) { return tableRule.get().getKeyGenerator().generateKey(); } return defaultKeyGenerator.generateKey(); } /** * Get logic table name base on logic index name. * * @param logicIndexName logic index name * @return logic table name */ public String getLogicTableName(final String logicIndexName) { for (TableRule each : tableRules) { if (logicIndexName.equals(each.getLogicIndex())) { return each.getLogicTable(); } } throw new ShardingConfigurationException("Cannot find logic table name with logic index name: '%s'", logicIndexName); } /** * Find data node by logic table. * * @param logicTableName logic table name * @return data node */ public DataNode findDataNode(final String logicTableName) { return findDataNode(null, logicTableName); } /** * Find data node by data source and logic table. * * @param dataSourceName data source name * @param logicTableName logic table name * @return data node */ public DataNode findDataNode(final String dataSourceName, final String logicTableName) { TableRule tableRule = getTableRuleByLogicTableName(logicTableName); for (DataNode each : tableRule.getActualDataNodes()) { if (shardingDataSourceNames.getDataSourceNames().contains(each.getDataSourceName()) && (null == dataSourceName || each.getDataSourceName().equals(dataSourceName))) { return each; } } if (null == dataSourceName) { throw new ShardingConfigurationException("Cannot find actual data node for logic table name: '%s'", logicTableName); } else { throw new ShardingConfigurationException("Cannot find actual data node for data source name: '%s' and logic table name: '%s'", dataSourceName, logicTableName); } } /** * Adjust is logic index or not. * * @param logicIndexName logic index name * @param logicTableName logic table name * @return is logic index or not */ public boolean isLogicIndex(final String logicIndexName, final String logicTableName) { return logicIndexName.equals(getTableRuleByLogicTableName(logicTableName).getLogicIndex()); } /** * Find actual default data source name. * *

If use master-slave rule, return master data source name.

* * @return actual default data source name */ public Optional findActualDefaultDataSourceName() { String result = shardingDataSourceNames.getDefaultDataSourceName(); if (Strings.isNullOrEmpty(result)) { return Optional.absent(); } return Optional.of(getMasterDataSourceName(result)); } private String getMasterDataSourceName(final String masterSlaveRuleName) { for (MasterSlaveRule each : masterSlaveRules) { if (each.getName().equals(masterSlaveRuleName)) { return each.getMasterDataSourceName(); } } return masterSlaveRuleName; } /** * Find master slave rule. * * @param dataSourceName data source name * @return master slave rule */ public Optional findMasterSlaveRule(final String dataSourceName) { for (MasterSlaveRule each : masterSlaveRules) { if (each.containDataSourceName(dataSourceName)) { return Optional.of(each); } } return Optional.absent(); } /** * Get actual data source name by actual table name. * * @param actualTableName actual table name * @return actual data source name */ public String getActualDataSourceNameByActualTableName(final String actualTableName) { Optional tableRule = tryFindTableRuleByActualTable(actualTableName); if (tableRule.isPresent()) { return tableRule.get().getActualDatasourceNames().iterator().next(); } if (!Strings.isNullOrEmpty(getShardingDataSourceNames().getDefaultDataSourceName())) { return getShardingDataSourceNames().getDefaultDataSourceName(); } throw new ShardingException("Cannot found actual data source name of '%s' in sharding rule.", actualTableName); } }




© 2015 - 2025 Weber Informatics LLC | Privacy Policy