Added process to add UUID column to table and generate uuid key for existing records.

This commit is contained in:
Heng Sin Low 2010-12-15 18:03:52 +08:00
parent 5bdee92457
commit 8816fa5a9f
1 changed files with 258 additions and 0 deletions

View File

@ -0,0 +1,258 @@
/******************************************************************************
* Product: Adempiere ERP & CRM Smart Business Solution *
* Copyright (C) 2010 Heng Sin Low *
* This program is free software; you can redistribute it and/or modify it *
* under the terms version 2 of the GNU General Public License as published *
* by the Free Software Foundation. This program is distributed in the hope *
* that it will be useful, but WITHOUT ANY WARRANTY; without even the implied *
* warranty of MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. *
* See the GNU General Public License for more details. *
* You should have received a copy of the GNU General Public License along *
* with this program; if not, write to the Free Software Foundation, Inc., *
* 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA. *
*****************************************************************************/
package org.adempiere.process;
import java.math.BigDecimal;
import java.sql.Connection;
import java.sql.DatabaseMetaData;
import java.sql.PreparedStatement;
import java.sql.ResultSet;
import java.sql.SQLException;
import java.util.UUID;
import org.adempiere.exceptions.DBException;
import org.compiere.model.MColumn;
import org.compiere.model.MTable;
import org.compiere.model.M_Element;
import org.compiere.process.ProcessInfoParameter;
import org.compiere.process.SvrProcess;
import org.compiere.util.AdempiereUserError;
import org.compiere.util.CLogger;
import org.compiere.util.DB;
import org.compiere.util.DisplayType;
import org.compiere.util.Trx;
import org.compiere.util.ValueNamePair;
/**
* Add UUID column (tableName_UU) to table and update existing records with new UUID.
* Warning: this process is only safe to run if it have exclusive access to database.
* @author hengsin
*
*/
public class UUIDGenerator extends SvrProcess {
private String tableName;
/**
* @see org.compiere.process.SvrProcess#prepare()
*/
@Override
protected void prepare() {
ProcessInfoParameter[] parameters = getProcessInfo().getParameter();
if (parameters == null || parameters.length == 0)
return;
for(ProcessInfoParameter param : parameters) {
if (param.getParameterName().equals("TableName")) {
tableName = param.getParameter().toString();
break;
}
}
}
/**
* @see org.compiere.process.SvrProcess#doIt()
*/
@Override
protected String doIt() throws Exception {
if (tableName == null || tableName.trim().length() == 0)
tableName = "%";
else
tableName = tableName.trim();
if (!tableName.endsWith("%"))
tableName = tableName + "%";
String sql = "SELECT AD_Table_ID, TableName FROM AD_Table WHERE TableName like ? AND IsView = 'N' AND IsActive='Y'";
PreparedStatement stmt = null;
ResultSet rs = null;
int count = 0;
try {
stmt = DB.prepareStatement(sql, null);
stmt.setString(1, tableName);
rs = stmt.executeQuery();
while(rs.next()) {
int AD_Table_ID = rs.getInt(1);
String cTableName = rs.getString(2);
//skip import and translation table
if (cTableName.startsWith("I_") || cTableName.endsWith("_Trl"))
continue;
String columnName = cTableName + "_UU";
int AD_Column_ID = DB.getSQLValue(null, "SELECT AD_Column_ID FROM AD_Column Where AD_Table_ID = ? AND ColumnName = ?", AD_Table_ID, columnName);
if (AD_Column_ID <= 0) {
System.out.println("Adding UUID to " + cTableName);
count++;
//create column
MColumn mColumn = new MColumn(getCtx(), 0, null);
mColumn.setAD_Table_ID(AD_Table_ID);
int AD_Element_ID = DB.getSQLValue(null, "SELECT AD_Element_ID FROM AD_Element WHERE ColumnName=?",columnName);
if (AD_Element_ID <= 0) {
M_Element adElement = new M_Element(getCtx(), 0, null);
adElement.setColumnName(columnName);
adElement.setName(columnName);
adElement.setPrintName(columnName);
adElement.setEntityType("U");
adElement.saveEx();
AD_Element_ID = adElement.getAD_Element_ID();
}
mColumn.setAD_Element_ID(AD_Element_ID);
mColumn.setColumnName(columnName);
mColumn.setAD_Reference_ID(DisplayType.String);
mColumn.setEntityType("U");
mColumn.setFieldLength(36);
mColumn.setName(columnName);
mColumn.setVersion(new BigDecimal("1.00"));
mColumn.saveEx();
syncColumn(mColumn);
//update db
updateUUID(mColumn);
}
}
} finally {
DB.close(rs,stmt);
}
return count + " table altered";
}
private void updateUUID(MColumn column) {
MTable table = (MTable) column.getAD_Table();
int AD_Column_ID = DB.getSQLValue(null, "SELECT AD_Column_ID FROM AD_Column WHERE AD_Table_ID=? AND ColumnName=?", table.getAD_Table_ID(), table.getTableName()+"_ID");
StringBuffer sql = new StringBuffer("SELECT ");
String keyColumn = null;
if (AD_Column_ID > 0) {
keyColumn = table.getTableName()+"_ID";
} else if (DB.isOracle()) {
keyColumn = "rowid";
} else if (DB.isPostgreSQL()) {
keyColumn = "ctid";
}
sql.append(keyColumn).append(" FROM ").append(table.getTableName());
String updateSQL = "UPDATE "+table.getTableName()+" SET "+column.getColumnName()+"=? WHERE "+keyColumn+"=";
PreparedStatement stmt = null;
ResultSet rs = null;
Trx trx = null;
try {
trx = Trx.get(Trx.createTrxName(), true);
trx.start();
stmt = DB.prepareStatement(sql.toString(), trx.getTrxName());
stmt.setFetchSize(100);
rs = stmt.executeQuery();
while (rs.next()) {
UUID uuid = UUID.randomUUID();
if (AD_Column_ID > 0) {
int recordId = rs.getInt(1);
DB.executeUpdateEx(updateSQL+recordId,new Object[]{uuid.toString()},null);
} else {
String rowId = rs.getString(1);
DB.executeUpdateEx(updateSQL+"'"+rowId+"'",new Object[]{uuid.toString()},null);
}
}
} catch (SQLException e) {
throw new DBException(e);
} finally {
DB.close(rs, stmt);
if (trx != null)
trx.close();
}
}
private void syncColumn(MColumn column) {
// Find Column in Database
Connection conn = null;
try {
conn = DB.getConnectionRO();
DatabaseMetaData md = conn.getMetaData();
String catalog = DB.getDatabase().getCatalog();
String schema = DB.getDatabase().getSchema();
MTable table = (MTable) column.getAD_Table();
String tableName = table.getTableName();
if (md.storesUpperCaseIdentifiers())
{
tableName = tableName.toUpperCase();
}
else if (md.storesLowerCaseIdentifiers())
{
tableName = tableName.toLowerCase();
}
int noColumns = 0;
String sql = null;
//
ResultSet rs = null;
try
{
rs = md.getColumns(catalog, schema, tableName, null);
while (rs.next())
{
noColumns++;
String columnName = rs.getString ("COLUMN_NAME");
if (!columnName.equalsIgnoreCase(column.getColumnName()))
continue;
// update existing column
boolean notNull = DatabaseMetaData.columnNoNulls == rs.getInt("NULLABLE");
sql = column.getSQLModify(table, column.isMandatory() != notNull);
break;
}
}
finally
{
DB.close(rs);
}
// No Table
if (noColumns == 0)
sql = table.getSQLCreate ();
// No existing column
else if (sql == null)
sql = column.getSQLAdd(table);
int no = 0;
if (sql.indexOf(DB.SQLSTATEMENT_SEPARATOR) == -1)
{
no = DB.executeUpdate(sql, false, null);
addLog (0, null, new BigDecimal(no), sql);
}
else
{
String statements[] = sql.split(DB.SQLSTATEMENT_SEPARATOR);
for (int i = 0; i < statements.length; i++)
{
int count = DB.executeUpdate(statements[i], false, null);
addLog (0, null, new BigDecimal(count), statements[i]);
no += count;
}
}
if (no == -1)
{
String msg = "@Error@ ";
ValueNamePair pp = CLogger.retrieveError();
if (pp != null)
msg = pp.getName() + " - ";
msg += sql;
throw new AdempiereUserError (msg);
}
//TODO: create unique index
} catch (SQLException e) {
throw new DBException(e);
} finally {
if (conn != null) {
try {
conn.close();
} catch (Exception e) {}
}
}
}
}