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

com.litongjava.db.activerecord.dialect.PostgreSqlDialect Maven / Gradle / Ivy

package com.litongjava.db.activerecord.dialect;

import java.math.BigInteger;
import java.sql.PreparedStatement;
import java.sql.ResultSet;
import java.sql.ResultSetMetaData;
import java.sql.SQLException;
import java.sql.Types;
import java.util.List;
import java.util.Map;
import java.util.Map.Entry;
import java.util.Set;

import org.postgresql.util.PGobject;

import com.litongjava.db.activerecord.CPI;
import com.litongjava.db.activerecord.Model;
import com.litongjava.db.activerecord.Record;
import com.litongjava.db.activerecord.Table;
import com.litongjava.db.activerecord.builder.TimestampProcessedModelBuilder;
import com.litongjava.db.activerecord.builder.TimestampProcessedRecordBuilder;
import com.litongjava.tio.utils.json.Json;

/**
 * PostgreSqlDialect.
 */
public class PostgreSqlDialect extends Dialect {

  public PostgreSqlDialect() {
    this.modelBuilder = TimestampProcessedModelBuilder.me;
    this.recordBuilder = TimestampProcessedRecordBuilder.me;
  }

  public String forTableBuilderDoBuild(String tableName) {
    return "select * from \"" + tableName + "\" where 1 = 2";
  }

  public String forFindAll(String tableName) {
    return "select * from \"" + tableName + "\"";
  }

  public void forModelSave(Table table, Map attrs, StringBuilder sql, List paras) {
    sql.append("insert into \"").append(table.getName()).append("\"(");
    StringBuilder temp = new StringBuilder(") values(");
    for (Entry e : attrs.entrySet()) {
      String colName = e.getKey();
      if (table.hasColumnLabel(colName)) {
        if (paras.size() > 0) {
          sql.append(", ");
          temp.append(", ");
        }
        sql.append('\"').append(colName).append('\"');
        temp.append('?');
        paras.add(e.getValue());
      }
    }
    sql.append(temp.toString()).append(')');
  }

  public String forModelDeleteById(Table table) {
    String[] pKeys = table.getPrimaryKey();
    StringBuilder sql = new StringBuilder(45);
    sql.append("delete from \"");
    sql.append(table.getName());
    sql.append("\" where ");
    for (int i = 0; i < pKeys.length; i++) {
      if (i > 0) {
        sql.append(" and ");
      }
      sql.append('\"').append(pKeys[i]).append("\" = ?");
    }
    return sql.toString();
  }

  public void forModelUpdate(Table table, Map attrs, Set modifyFlag, StringBuilder sql, List paras) {
    sql.append("update \"").append(table.getName()).append("\" set ");
    String[] pKeys = table.getPrimaryKey();
    for (Entry e : attrs.entrySet()) {
      String colName = e.getKey();
      if (modifyFlag.contains(colName) && !isPrimaryKey(colName, pKeys) && table.hasColumnLabel(colName)) {
        if (paras.size() > 0) {
          sql.append(", ");
        }
        sql.append('\"').append(colName).append("\" = ? ");
        paras.add(e.getValue());
      }
    }
    sql.append(" where ");
    for (int i = 0; i < pKeys.length; i++) {
      if (i > 0) {
        sql.append(" and ");
      }
      sql.append('\"').append(pKeys[i]).append("\" = ?");
      paras.add(attrs.get(pKeys[i]));
    }
  }

  public String forModelFindById(Table table, String columns) {
    StringBuilder sql = new StringBuilder("select ");
    columns = columns.trim();
    if ("*".equals(columns)) {
      sql.append('*');
    } else {
      String[] arr = columns.split(",");
      for (int i = 0; i < arr.length; i++) {
        if (i > 0) {
          sql.append(',');
        }
        sql.append('\"').append(arr[i].trim()).append('\"');
      }
    }

    sql.append(" from \"");
    sql.append(table.getName());
    sql.append("\" where ");
    String[] pKeys = table.getPrimaryKey();
    for (int i = 0; i < pKeys.length; i++) {
      if (i > 0) {
        sql.append(" and ");
      }
      sql.append('\"').append(pKeys[i]).append("\" = ?");
    }
    return sql.toString();
  }

  public String forDbFindById(String tableName, String[] pKeys) {
    tableName = tableName.trim();
    trimPrimaryKeys(pKeys);

    StringBuilder sql = new StringBuilder("select * from \"").append(tableName).append("\" where ");
    for (int i = 0; i < pKeys.length; i++) {
      if (i > 0) {
        sql.append(" and ");
      }
      sql.append('\"').append(pKeys[i]).append("\" = ?");
    }
    return sql.toString();
  }

  @Override
  public void forDbDelete(String tableName, String[] pKeys, Record record, StringBuilder sql, List paras) {
    tableName = tableName.trim();
    trimPrimaryKeys(pKeys); // important

    sql.append("delete from \"");
    sql.append(tableName).append("\"");

    sql.append(" where ");

    int i = 0;
    for (Entry e : record.getColumns().entrySet()) {
      if (i > 0) {
        sql.append(" and ");
      }
      sql.append('\"').append(e.getKey()).append('\"').append("=? ");
      paras.add(e.getValue());
      i++;
    }
  }

  public String forDbDeleteById(String tableName, String[] pKeys) {
    tableName = tableName.trim();
    trimPrimaryKeys(pKeys);

    StringBuilder sql = new StringBuilder("delete from \"").append(tableName).append("\" where ");
    for (int i = 0; i < pKeys.length; i++) {
      if (i > 0) {
        sql.append(" and ");
      }
      sql.append('\"').append(pKeys[i]).append("\" = ?");
    }
    return sql.toString();
  }

  public void forDbSave(String tableName, String[] pKeys, Record record, StringBuilder sql, List paras) {
    tableName = tableName.trim();
    trimPrimaryKeys(pKeys);

    sql.append("insert into \"");
    sql.append(tableName).append("\"(");
    StringBuilder temp = new StringBuilder();
    temp.append(") values(");

    for (Entry e : record.getColumns().entrySet()) {
      if (paras.size() > 0) {
        sql.append(", ");
        temp.append(", ");
      }
      sql.append('\"').append(e.getKey()).append('\"');
      temp.append('?');
      paras.add(e.getValue());
    }
    sql.append(temp.toString()).append(')');
  }

  public void forDbUpdate(String tableName, String[] pKeys, Object[] ids, Record record, StringBuilder sql, List paras) {
    tableName = tableName.trim();
    trimPrimaryKeys(pKeys);

    // Record 新增支持 modifyFlag
    Set modifyFlag = CPI.getModifyFlag(record);

    sql.append("update \"").append(tableName).append("\" set ");
    for (Entry e : record.getColumns().entrySet()) {
      String colName = e.getKey();
      if (modifyFlag.contains(colName) && !isPrimaryKey(colName, pKeys)) {
        if (paras.size() > 0) {
          sql.append(", ");
        }
        sql.append('\"').append(colName).append("\" = ? ");
        paras.add(e.getValue());
      }
    }
    sql.append(" where ");
    for (int i = 0; i < pKeys.length; i++) {
      if (i > 0) {
        sql.append(" and ");
      }
      sql.append('\"').append(pKeys[i]).append("\" = ?");
      paras.add(ids[i]);
    }
  }

  public String forPaginate(int pageNumber, int pageSize, StringBuilder findSql) {
    int offset = pageSize * (pageNumber - 1);
    findSql.append(" limit ").append(pageSize).append(" offset ").append(offset);
    return findSql.toString();
  }

  public void fillStatement(PreparedStatement pst, List paras) throws SQLException {
    fillStatementHandleDateType(pst, paras);
  }

  public void fillStatement(PreparedStatement pst, Object... paras) throws SQLException {
    fillStatementHandleDateType(pst, paras);
  }

  protected void fillStatementHandleDateType(PreparedStatement pst, Object... paras) throws SQLException {
    for (int i = 0, size = paras.length; i < size; i++) {
      Object value = paras[i];
      if (value instanceof java.util.Date) {
        if (value instanceof java.sql.Date) {
          pst.setDate(i + 1, (java.sql.Date) value);
        } else if (value instanceof java.sql.Timestamp) {
          pst.setTimestamp(i + 1, (java.sql.Timestamp) value);
        } else {
          // Oracle、SqlServer 中的 TIMESTAMP、DATE 支持 new Date() 给值
          java.util.Date d = (java.util.Date) value;
          pst.setTimestamp(i + 1, new java.sql.Timestamp(d.getTime()));
        }
      } else if (value instanceof PGobject) {
        pst.setObject(i + 1, value);
      } else {
        if (value instanceof String) {
          String jsonValue = (String) value;
          if (jsonValue.startsWith("{") || jsonValue.startsWith("[")) {
            // add support for json
            pst.setObject(i + 1, value, Types.OTHER);
          } else {
            pst.setObject(i + 1, value);
          }
        } else {
          pst.setObject(i + 1, value);
        }
      }
    }
  }

  /**
   * 解决 PostgreSql 获取自增主键时 rs.getObject(1) 总是返回第一个字段的值,而非返回了 id 值 issue:
   * https://www.oschina.net/question/2312705_2243354
   * 

* 相对于 Dialect 中的默认实现,仅将 rs.getXxx(1) 改成了 rs.getXxx(pKey) */ public void getModelGeneratedKey(Model model, PreparedStatement pst, Table table) throws SQLException { String[] pKeys = table.getPrimaryKey(); ResultSet rs = pst.getGeneratedKeys(); for (String pKey : pKeys) { if (model.get(pKey) == null || isOracle()) { if (rs.next()) { Class colType = table.getColumnType(pKey); if (colType != null) { if (colType == Integer.class || colType == int.class) { model.set(pKey, rs.getInt(pKey)); } else if (colType == Long.class || colType == long.class) { model.set(pKey, rs.getLong(pKey)); } else if (colType == BigInteger.class) { processGeneratedBigIntegerKey(model, pKey, rs.getObject(pKey)); } else { model.set(pKey, rs.getObject(pKey)); } } } } } rs.close(); } /** * 解决 PostgreSql 获取自增主键时 rs.getObject(1) 总是返回第一个字段的值,而非返回了 id 值 issue: * https://www.oschina.net/question/2312705_2243354 *

* 相对于 Dialect 中的默认实现,仅将 rs.getXxx(1) 改成了 rs.getXxx(pKey) */ public void getRecordGeneratedKey(PreparedStatement pst, Record record, String[] pKeys) { try (ResultSet rs = pst.getGeneratedKeys()) { // Automatically closes ResultSet ResultSetMetaData metaData = rs.getMetaData(); int columnCount = metaData.getColumnCount(); if (rs.next()) { for (int i = 1; i <= columnCount; i++) { String name = metaData.getColumnName(i); int columnType = metaData.getColumnType(i); Object value; switch (columnType) { case java.sql.Types.SMALLINT: value = rs.getShort(i); break; case java.sql.Types.INTEGER: value = rs.getInt(i); break; default: value = rs.getObject(i); break; } record.set(name, value); } } } catch (SQLException e) { throw new RuntimeException(e); } } @Override public String forDbFindColumnsById(String tableName, String columns, String[] pKeys) { return DialectUtils.forDbFindColumnsById(tableName, columns, pKeys); } @Override public String forDbFindColumns(String tableName, String columns) { return DialectUtils.forDbFindColumns(tableName, columns); } @Override public String forExistsByFields(String tableName, String fields) { StringBuffer stringBuffer = new StringBuffer(); stringBuffer.append("select count(1) from \"").append(tableName).append("\""); String[] split = fields.split(","); if (split.length > 0) { stringBuffer.append(" where "); for (String field : split) { stringBuffer.append('\"').append(field.trim()).append('\"').append("= ?"); } } return stringBuffer.toString(); } @Override public void forDbUpdate(String tableName, String[] pKeys, Object[] ids, Record record, StringBuilder sql, List paras, String[] jsonFields) { if (jsonFields != null && jsonFields.length > 0) { for (String f : jsonFields) { Object object = record.get(f); if (object != null) { PGobject pGobject = new PGobject(); pGobject.setType("json"); String jsonString = null; if (object instanceof String) { jsonString = (String) object; if ("".equals(jsonString)) { try { pGobject.setValue(null); record.set(f, pGobject); } catch (SQLException e) { throw new RuntimeException(e); } } else if (jsonString.startsWith("\"") && jsonString.endsWith("\"")) { try { pGobject.setValue(jsonString); record.set(f, pGobject); } catch (SQLException e) { throw new RuntimeException(e); } } else if (jsonString.startsWith("{") && jsonString.endsWith("}")) { try { pGobject.setValue(jsonString); record.set(f, pGobject); } catch (SQLException e) { throw new RuntimeException(e); } } else if (jsonString.startsWith("[") && jsonString.endsWith("]")) { try { pGobject.setValue(jsonString); record.set(f, pGobject); } catch (SQLException e) { throw new RuntimeException(e); } } } else { jsonString = Json.getJson().toJson(object); try { pGobject.setValue(jsonString); record.set(f, pGobject); } catch (SQLException e) { throw new RuntimeException(e); } } } } } forDbUpdate(tableName, pKeys, ids, record, sql, paras); } @Override public void forDbSave(String tableName, String[] pKeys, Record record, StringBuilder sql, List paras, String[] jsonFields) { if (jsonFields != null && jsonFields.length > 0) { for (String f : jsonFields) { Object object = record.get(f); if (object != null) { PGobject pGobject = new PGobject(); pGobject.setType("json"); String jsonString = null; if (object instanceof String) { jsonString = (String) object; } else { jsonString = Json.getJson().toJson(object); } try { pGobject.setValue(jsonString); record.set(f, pGobject); } catch (SQLException e) { throw new RuntimeException(e); } } } } this.forDbSave(tableName, pKeys, record, sql, paras); } }