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

com.litongjava.db.activerecord.ReplicaActiveRecordPlugin Maven / Gradle / Ivy

There is a newer version: 1.4.8
Show newest version
package com.litongjava.db.activerecord;

import java.sql.Connection;
import java.util.ArrayList;
import java.util.List;

import javax.sql.DataSource;

import com.jfinal.kit.StrKit;
import com.litongjava.cache.ICache;
import com.litongjava.db.IPlugin;
import com.litongjava.db.activerecord.cache.DefaultEhCache;
import com.litongjava.db.activerecord.dialect.Dialect;
import com.litongjava.db.activerecord.dialect.MysqlDialect;
import com.litongjava.db.activerecord.sql.SqlKit;

/**
 * ActiveRecord plugin. 
* ActiveRecord plugin not support mysql type year, you can use int instead of * year. Mysql error message for type year when insert a record: Data truncated * for column 'xxx' at row 1 */ public class ReplicaActiveRecordPlugin implements IPlugin { protected TableBuilder tableBuilder = new TableBuilder(); protected List dataSourceProviders; protected Boolean devMode = null; protected List configs; protected volatile boolean isStarted = false; protected List tableList = new ArrayList
(); public ReplicaActiveRecordPlugin(List datasources) { this.configs = new ArrayList<>(); for (int i = 0; i < datasources.size(); i++) { this.configs.add(new Config(DbKit.REPLICA_CONFIG_NAME + "_" + i, datasources.get(i), DbKit.DEFAULT_TRANSACTION_LEVEL)); } } public ReplicaActiveRecordPlugin(String configName, DataSource dataSource, int transactionLevel) { this.configs = new ArrayList<>(); if (StrKit.isBlank(configName)) { throw new IllegalArgumentException("configName can not be blank"); } if (dataSource == null) { throw new IllegalArgumentException("dataSource can not be null"); } this.configs.add(new Config(configName, dataSource, transactionLevel)); } public ReplicaActiveRecordPlugin(DataSource dataSource) { this(DbKit.MAIN_CONFIG_NAME, dataSource); } public ReplicaActiveRecordPlugin(String configName, DataSource dataSource) { this(configName, dataSource, DbKit.DEFAULT_TRANSACTION_LEVEL); } public ReplicaActiveRecordPlugin(DataSource dataSource, int transactionLevel) { this(DbKit.MAIN_CONFIG_NAME, dataSource, transactionLevel); } public ReplicaActiveRecordPlugin(String configName, IDataSourceProvider dataSourceProvider, int transactionLevel) { this.dataSourceProviders=new ArrayList<>(); if (StrKit.isBlank(configName)) { throw new IllegalArgumentException("configName can not be blank"); } if (dataSourceProvider == null) { throw new IllegalArgumentException("dataSourceProvider can not be null"); } this.dataSourceProviders.add(dataSourceProvider); this.configs.add(new Config(configName, null, transactionLevel)); } public ReplicaActiveRecordPlugin(String configName, List dataSourceProviders, int transactionLevel) { this.dataSourceProviders=new ArrayList<>(); if (StrKit.isBlank(configName)) { throw new IllegalArgumentException("configName can not be blank"); } if (dataSourceProviders == null) { throw new IllegalArgumentException("dataSourceProviders can not be null"); } for (int i = 0; i < dataSourceProviders.size(); i++) { this.dataSourceProviders.add(dataSourceProviders.get(i)); this.configs.add(new Config(configName + "_" + i, null, transactionLevel)); } } public ReplicaActiveRecordPlugin(IDataSourceProvider dataSourceProvider) { this(DbKit.REPLICA_CONFIG_NAME, dataSourceProvider); } public ReplicaActiveRecordPlugin(String configName, IDataSourceProvider dataSourceProvider) { this(configName, dataSourceProvider, DbKit.DEFAULT_TRANSACTION_LEVEL); } public ReplicaActiveRecordPlugin(IDataSourceProvider dataSourceProvider, int transactionLevel) { this(DbKit.REPLICA_CONFIG_NAME, dataSourceProvider, transactionLevel); } public ReplicaActiveRecordPlugin(Config config) { if (config == null) { throw new IllegalArgumentException("Config can not be null"); } this.configs.add(config); } public ReplicaActiveRecordPlugin addMapping(String tableName, String primaryKey, Class> modelClass) { tableList.add(new Table(tableName, primaryKey, modelClass)); return this; } public ReplicaActiveRecordPlugin addMapping(String tableName, Class> modelClass) { tableList.add(new Table(tableName, modelClass)); return this; } public ReplicaActiveRecordPlugin addSqlTemplate(String sqlTemplate) { for (Config config : configs) { config.sqlKit.addSqlTemplate(sqlTemplate); } return this; } public ReplicaActiveRecordPlugin addSqlTemplate(com.jfinal.template.source.ISource sqlTemplate) { for (Config config : configs) { config.sqlKit.addSqlTemplate(sqlTemplate); } return this; } public ReplicaActiveRecordPlugin setBaseSqlTemplatePath(String baseSqlTemplatePath) { for (Config config : configs) { config.sqlKit.setBaseSqlTemplatePath(baseSqlTemplatePath); } return this; } public SqlKit getSqlKit(int i) { return configs.get(i).sqlKit; } public com.jfinal.template.Engine getEngine(int i) { return getSqlKit(i).getEngine(); } /** * Set transaction level define in java.sql.Connection * * @param transactionLevel only be 0, 1, 2, 4, 8 */ public ReplicaActiveRecordPlugin setTransactionLevel(int transactionLevel) { for (Config config : configs) { config.setTransactionLevel(transactionLevel); } return this; } public ReplicaActiveRecordPlugin setCache(ICache cache) { if (cache == null) { throw new IllegalArgumentException("cache can not be null"); } for (Config config : configs) { config.cache = cache; } return this; } public ReplicaActiveRecordPlugin setShowSql(boolean showSql) { for (Config config : configs) { config.showSql = showSql; } return this; } public ReplicaActiveRecordPlugin setDevMode(boolean devMode) { this.devMode = devMode; for (Config config : configs) { config.setDevMode(devMode); } return this; } public Boolean getDevMode() { return devMode; } public ReplicaActiveRecordPlugin setDialect(Dialect dialect) { if (dialect == null) { throw new IllegalArgumentException("dialect can not be null"); } for (Config config : configs) { config.dialect = dialect; if (config.transactionLevel == Connection.TRANSACTION_REPEATABLE_READ && dialect.isOracle()) { // Oracle 不支持 Connection.TRANSACTION_REPEATABLE_READ config.transactionLevel = Connection.TRANSACTION_READ_COMMITTED; } } return this; } public ReplicaActiveRecordPlugin setContainerFactory(IContainerFactory containerFactory) { if (containerFactory == null) { throw new IllegalArgumentException("containerFactory can not be null"); } for (Config config : configs) { config.containerFactory = containerFactory; } return this; } public ReplicaActiveRecordPlugin setDbProFactory(IDbProFactory dbProFactory) { if (dbProFactory == null) { throw new IllegalArgumentException("dbProFactory can not be null"); } for (Config config : configs) { config.dbProFactory = dbProFactory; } return this; } /** * 当使用 create table 语句创建用于开发使用的数据表副本时,假如create table 中使用的 * 复合主键次序不同,那么MappingKitGeneretor 反射生成的复合主键次序也会不同。 * * 而程序中类似于 model.deleteById(id1, id2) 方法中复合主键次序与需要与映射时的次序 保持一致,可以在MappingKit * 映射完成以后通过调用此方法再次强制指定复合主键次序 * *
   * Example:
   * ActiveRecrodPlugin arp = new ActiveRecordPlugin(...);
   * _MappingKit.mapping(arp);
   * arp.setPrimaryKey("account_role", "account_id, role_id");
   * me.add(arp);
   * 
*/ public void setPrimaryKey(String tableName, String primaryKey) { for (Table table : tableList) { if (table.getName().equalsIgnoreCase(tableName.trim())) { table.setPrimaryKey(primaryKey); } } } public boolean start() { if (isStarted) { return true; } for (int i = 0; i < configs.size(); i++) { Config config = configs.get(i); if (dataSourceProviders != null) { IDataSourceProvider dataSourceProvider = dataSourceProviders.get(i); if (config.dataSource == null && dataSourceProvider != null) { config.dataSource = dataSourceProvider.getDataSource(); } } if (config.dataSource == null) { throw new RuntimeException("ActiveRecord start error: ActiveRecordPlugin need DataSource or DataSourceProvider"); } config.sqlKit.parseSqlTemplate(); tableBuilder.build(tableList, config); } DbKit.addReplicaConfigs(configs); isStarted = true; return true; } public boolean stop() { for (Config config : configs) { DbKit.removeConfig(config.getName()); } isStarted = false; return true; } /** * 用于分布式场景,当某个分布式节点只需要用 Model 承载和传输数据,而不需要实际操作数据库时 调用本方法可保障 * IContainerFactory、Dialect、ICache 的一致性 * * 本用法更加适用于 Generator 生成的继承自 base model的 Model,更加便于传统第三方工具对 带有 getter、setter 的 * model 进行各种处理 * *
   * 警告:Dialect、IContainerFactory、ICache 三者一定要与集群中其它节点保持一致,
   *     以免程序出现不一致行为
   * 
*/ public static void useAsDataTransfer(Dialect dialect, IContainerFactory containerFactory, ICache cache) { if (dialect == null) { throw new IllegalArgumentException("dialect can not be null"); } if (containerFactory == null) { throw new IllegalArgumentException("containerFactory can not be null"); } if (cache == null) { throw new IllegalArgumentException("cache can not be null"); } ReplicaActiveRecordPlugin arp = new ReplicaActiveRecordPlugin(new NullDataSource()); arp.setDialect(dialect); arp.setContainerFactory(containerFactory); arp.setCache(cache); arp.start(); DbKit.brokenConfigs = arp.configs; } /** * 分布式场景下指定 IContainerFactory,并默认使用 MysqlDialect、EhCache * * @see #useAsDataTransfer(Dialect, IContainerFactory, IDbCache) */ public static void useAsDataTransfer(IContainerFactory containerFactory) { useAsDataTransfer(new com.litongjava.db.activerecord.dialect.MysqlDialect(), containerFactory, new DefaultEhCache()); } /** * 分布式场景下指定 Dialect、IContainerFactory,并默认使用 EhCache * * @see #useAsDataTransfer(Dialect, IContainerFactory, IDbCache) */ public static void useAsDataTransfer(Dialect dialect, IContainerFactory containerFactory) { useAsDataTransfer(dialect, containerFactory, new DefaultEhCache()); } /** * 分布式场景下指定 Dialect、 并默认使用 IContainerFactory.defaultContainerFactory、EhCache * * @see #useAsDataTransfer(Dialect, IContainerFactory, IDbCache) */ public static void useAsDataTransfer(Dialect dialect) { useAsDataTransfer(dialect, IContainerFactory.defaultContainerFactory, new DefaultEhCache()); } /** * 分布式场景下默认使用 MysqlDialect、 IContainerFactory.defaultContainerFactory、EhCache * * @see #useAsDataTransfer(Dialect, IContainerFactory, IDbCache) */ public static void useAsDataTransfer() { useAsDataTransfer(new MysqlDialect(), IContainerFactory.defaultContainerFactory, new DefaultEhCache()); } public Config getConfig(int i) { return configs.get(i); } /** * 一般用于配置 TableBuilder 内的 JavaType * *
   * 例如:
   *    ActiveRecordPlugin arp = ...;
   *    JavaType jt = arp.getTableBuilder().getJavaType();
   *    
   *    jt.addType(org.postgresql.geometric.PGpoint.class);
   *    jt.addType(org.postgresql.geometric.PGbox.class);
   *    jt.addType(org.postgresql.geometric.PGcircle.class);
   *    jt.addType(org.postgresql.geometric.PGline.class);
   *    jt.addType(org.postgresql.geometric.PGlseg.class);
   *    jt.addType(org.postgresql.geometric.PGpath.class);
   *    jt.addType(org.postgresql.geometric.PGpolygon.class);
   * 
*/ public TableBuilder getTableBuilder() { return tableBuilder; } /** * 可用于切换 TableBuilder 实现类 */ public ReplicaActiveRecordPlugin setTableBuilder(TableBuilder tableBuilder) { this.tableBuilder = tableBuilder; return this; } }