OracleService類

using System;
using System.Collections.Generic;
using System.Data;
using System.IO;
using System.Linq;
using System.Net.Mail;
using System.Net.Mime;
using System.ServiceModel;
using System.ServiceModel.Channels;
using System.Text;
using System.Threading;
using System.Web;
using System.Web.Hosting;
using Oracle.DataAccess.Client;
using Oracle.DataAccess.Types;

public class OracleService : IOracle
{
    protected OracleConnection con;
    protected Dictionary<string, DataTable> schemaTables;
    protected Dictionary<string, DataTable> pkTables;
    protected Circumstances SapEenvironment;

    
    public OracleService()
    {
        con = new OracleConnection();
    }
    public StoredProcedureParameter[] GetStoredProcedureValues(string procedureName, StoredProcedureParameter[] spps)
    {
        OracleCommand command = new OracleCommand(procedureName, con);
        command.CommandTimeout = 600;
        command.CommandType = CommandType.StoredProcedure;
        command.BindByName = true;
        for (int i = 0; i < spps.Length; i++)
        {
            OracleParameter parameter = command.Parameters.Add(spps[i].Name,
                getDbType(spps[i].OracleType), getDbDirection(spps[i].Direction));
            if (spps[i].Direction == DbDirection.Input || spps[i].Direction == DbDirection.InputOutput)
            {
                parameter.Value = spps[i].Value;
            }
        }

        OracleDataAdapter oda = new OracleDataAdapter();
        try
        {
            con.Open();
            int result = command.ExecuteNonQuery();
            for (int i = 0; i < spps.Length; i++)
            {
                DataSet ds = new DataSet();
                if (spps[i].Direction == DbDirection.Output || spps[i].Direction == DbDirection.InputOutput ||
                    spps[i].Direction == DbDirection.ReturnValue)
                {
                    if (spps[i].OracleType == DbType.RefCursor)
                    {
                        OracleRefCursor orc = (OracleRefCursor)command.Parameters[spps[i].Name].Value;
                        oda.Fill(ds, orc);
                        spps[i].Cursor = ds;
                        spps[i].Value = null;
                    }
                    else
                    {
                        spps[i].Value = command.Parameters[spps[i].Name].Value.ToString();
                        spps[i].Cursor = null;
                    }
                }
            }
            return spps;
        }
        catch (Exception ex)
        {
            writeLog(ex.Source + "\t" + ex.Message + "\r\n\t\t\t" + procedureName + "\r\n");
            throw ex;
        }
        finally
        {
            oda.Dispose();
            command.Dispose();
            con.Close();
        }
    }

    private ParameterDirection getDbDirection(DbDirection direction)
    {
        ParameterDirection pd;
        switch (direction)
        {
            case DbDirection.Input:
                pd = ParameterDirection.Input;
                break;
            case DbDirection.InputOutput:
                pd = ParameterDirection.InputOutput;
                break;
            case DbDirection.Output:
                pd = ParameterDirection.Output;
                break;
            case DbDirection.ReturnValue:
                pd = ParameterDirection.ReturnValue;
                break;
            default:
                pd = ParameterDirection.InputOutput;
                break;
        }
        return pd;
    }

    private OracleDbType getDbType(DbType dbType)
    {
        OracleDbType odt;
        switch (dbType)
        {
            case DbType.BFile:
                {
                    odt = OracleDbType.BFile;
                }
                break;
            case DbType.Blob:
                {
                    odt = OracleDbType.Blob;
                }
                break;
            case DbType.Byte:
                {
                    odt = OracleDbType.Byte;
                }
                break;
            case DbType.Char:
                {
                    odt = OracleDbType.Char;
                }
                break;
            case DbType.Clob:
                {
                    odt = OracleDbType.Clob;
                }
                break;
            case DbType.Date:
                {
                    odt = OracleDbType.Date;
                }
                break;
            case DbType.Decimal:
                {
                    odt = OracleDbType.Decimal;
                }
                break;
            case DbType.Double:
                {
                    odt = OracleDbType.Double;
                }
                break;
            case DbType.Int16:
                {
                    odt = OracleDbType.Int16;
                }
                break;
            case DbType.Int32:
                {
                    odt = OracleDbType.Int32;
                }
                break;
            case DbType.Int64:
                {
                    odt = OracleDbType.Int64;
                }
                break;
            case DbType.IntervalDS:
                {
                    odt = OracleDbType.IntervalDS;
                }
                break;
            case DbType.IntervalYM:
                {
                    odt = OracleDbType.IntervalYM;
                }
                break;
            case DbType.Long:
                {
                    odt = OracleDbType.Long;
                }
                break;
            case DbType.LongRaw:
                {
                    odt = OracleDbType.LongRaw;
                }
                break;
            case DbType.NChar:
                {
                    odt = OracleDbType.NChar;
                }
                break;
            case DbType.NClob:
                {
                    odt = OracleDbType.NClob;
                }
                break;
            case DbType.NVarchar2:
                {
                    odt = OracleDbType.NVarchar2;
                }
                break;
            case DbType.RefCursor:
                {
                    odt = OracleDbType.RefCursor;
                }
                break;
            case DbType.Single:
                {
                    odt = OracleDbType.Single;
                }
                break;
            case DbType.TimeStamp:
                {
                    odt = OracleDbType.TimeStamp;
                }
                break;
            case DbType.TimeStampLTZ:
                {
                    odt = OracleDbType.TimeStampLTZ;
                }
                break;
            case DbType.TimeStampTZ:
                {
                    odt = OracleDbType.TimeStampTZ;
                }
                break;
            case DbType.Varchar2:
                {
                    odt = OracleDbType.Varchar2;
                }
                break;
            default:
                odt = OracleDbType.Varchar2;
                break;
        }
        return odt;
    }

    public int ExecuteStroedProcedure(string procedureName, string[] args, string[] argValues)
    {
        OracleCommand command = new OracleCommand(procedureName, con);
        command.CommandTimeout = 600;
        command.CommandType = CommandType.StoredProcedure;

        for (int i = 0; i < args.Length; i++)
        {
            OracleParameter parameter = command.Parameters.Add(args[i], OracleDbType.Varchar2);
            parameter.Value = argValues[i];
        }

        try
        {
            con.Open();
            int result = command.ExecuteNonQuery();
            return result;
        }
        catch (Exception ex)
        {
            writeLog(ex.Source + "\t" + ex.Message + "\r\n\t\t\t" + procedureName + "\r\n");
            throw ex;
        }
        finally
        {
            command.Dispose();
            con.Close();
        }
    }

    public int ExecuteNonQuery(string SQL)
    {
        writeLog(SQL);
        OracleCommand command = new OracleCommand(SQL, con);
        OracleTransaction trans = null;
        try
        {
            con.Open();
            trans = con.BeginTransaction();
            int result = command.ExecuteNonQuery();
            trans.Commit();
            return result;
        }
        catch (Exception ex)
        {
            trans.Rollback();
            writeLog(ex.Source + "\t" + ex.Message + "\r\n\t\t\t" + SQL + "\r\n");
            throw ex;
        }
        finally
        {
            command.Dispose();
            con.Close();
        }
    }

    public void ExecuteBatchNonQuery(string[] SQL)
    {
        writeLog(String.Join("\r\n", SQL));
        OracleCommand command = new OracleCommand();
        command.Connection = con;
        con.Open();
        OracleTransaction trans = con.BeginTransaction();

        int index = 0;
        try
        {
            foreach(string sql in SQL)
            {
                command.CommandText = sql;
                command.ExecuteNonQuery();
                index++;
            }
            trans.Commit();
        }
        catch (Exception ex)
        {
            trans.Rollback();
            writeLog("Error Index:" + index.ToString() + "\r\n" + ex.Source + "\t" + 
                ex.Message + "\r\n\t\t\t" + command.CommandText + "\r\n");
            throw ex;
        }
        finally
        {
            command.Dispose();
            con.Close();
        }
    }

    public DataSet ExecuteBatchQuery(string[] SQL)
    {
        OracleCommand command = new OracleCommand();
        int index = 0;
        try
        {
            DataSet ds = new DataSet();
            command.Connection = con;
            foreach (string sql in SQL)
            {
                command.CommandText = sql;
                OracleDataAdapter da = new OracleDataAdapter(command);
                da.Fill(ds, index.ToString());
                index++;
            }

            return ds;
        }
        catch (Exception ex)
        {
            writeLog("Error Index:" + index.ToString() + "\r\n" + ex.Source + "\t" + 
                ex.Message + "\r\n\t\t\t" + command.CommandText + "\r\n");
            throw ex;
        }
        finally
        {
            command.Dispose();
        }
    }

    public DataSet ExecuteQuery(string SQL)
    {
        DataSet ds = new DataSet();
        OracleCommand command = new OracleCommand();
        try
        {
            string[] temp = SQL.Split(";".ToCharArray());
            command.Connection = con;
            for (int i = 0; i < temp.Length; i++)
            {
                command.CommandText = temp[i];
                OracleDataAdapter da = new OracleDataAdapter(command);
                da.Fill(ds, i.ToString());
            }
            return ds;
        }
        catch (Exception ex)
        {
            writeLog(ex.Source + "\t" + ex.Message + "\n" + SQL + "\r\n");
            throw ex;
        }
        finally
        {
            command.Dispose();
        }
    }

    //單個SQL,及變量參數
    public DataSet ExecuteParameterQuery(string SQL, string[] parameterValues)
    {
        DataSet ds = new DataSet();
        OracleCommand command = new OracleCommand(SQL, con);
        foreach (string data in parameterValues)
        {
            OracleParameter parameter = new OracleParameter();
            parameter.Value = data;
            command.Parameters.Add(parameter);
        }
        try
        {
            OracleDataAdapter da = new OracleDataAdapter(command);
            da.Fill(ds, "0");
            return ds;
        }
        catch (Exception ex)
        {
            writeLog(ex.Source + "\t" + ex.Message + "\r\n\t\t\t" + SQL + "\r\n");
            throw ex;
        }
        finally
        {
            command.Dispose();
        }
    }

    //多個SQL,及變量參數
    public DataSet ExecuteBatchParameterQuery(string[] SQL, string[][] parameterValues)
    {
        DataSet ds = new DataSet();
        OracleCommand command = new OracleCommand();

        int index = 0;
        try
        {
            command.Connection = con;
            foreach (string sql in SQL)
            {
                command.CommandText = sql;
                command.Parameters.Clear();
                foreach (string data in parameterValues[index])
                {
                    OracleParameter parameter = new OracleParameter();
                    parameter.Value = data;
                    command.Parameters.Add(parameter);
                }
                OracleDataAdapter da = new OracleDataAdapter(command);
                da.Fill(ds, index.ToString());
                index++;
            }
            return ds;
        }
        catch (Exception ex)
        {
            writeLog("Error Index:" + index.ToString() + "\r\n" + ex.Source + "\t" +
                ex.Message + "\r\n\t\t\t" + command.CommandText + "\r\n");
            throw ex;
        }
        finally
        {
            command.Dispose();
        }
    }

    public string[] GetFieldData(string SQL)
    {
        List<string> list = new List<string>();
        OracleCommand command = new OracleCommand(SQL, con);
        try
        {
            con.Open();
            OracleDataReader dr = command.ExecuteReader();
            while (dr.Read())
            {
                list.Add(dr[0].ToString());
            }
            dr.Close();
            return list.ToArray();
        }
        catch (Exception ex)
        {
            writeLog(ex.Source + "\t" + ex.Message + "\r\n\t\t\t" + SQL + "\r\n");
            throw ex;
        }
        finally
        {
            command.Dispose();
            con.Close();
        }
    }

    public string GetSingleData(string SQL)
    {
        object temp = null;
        OracleCommand command = new OracleCommand(SQL, con);
        try
        {
            con.Open();
            temp = command.ExecuteScalar();
            return temp == null ? null : temp.ToString();
        }
        catch (Exception ex)
        {
            writeLog(ex.Source + "\t" + ex.Message + "\r\n\t\t\t" + SQL + "\r\n");
            throw ex;
        }
        finally
        {
            command.Dispose();
            con.Close();
        }
    }

    private object convertParameterValue(OracleParameter op, object strValue)
    {
        object temp = null;
        switch (op.OracleDbType)
        {
            case OracleDbType.Date:
                temp = Convert.ToDateTime(strValue.ToString());
                break;
            case OracleDbType.TimeStamp:
                temp = Convert.ToDateTime(strValue.ToString());
                break;
            case OracleDbType.Blob:
                temp = Convert.FromBase64String(strValue.ToString());
                break;
            default:
                temp = strValue;
                break;
        }
        return temp;
    }

    public int ExecuteUpdate(string tableName, string[] fields, string[] fieldValues, string[] whereFields, string[] whereValues)
    {
        StringBuilder sb = new StringBuilder();
        bool isSbOk = toStringBuilder(sb, "UPDATE", tableName, fields, fieldValues, whereFields, whereValues);
        writeLog(isSbOk.ToString() + "\r\n" + sb.ToString());

        List<string> listTemp = new List<string>();
        listTemp.AddRange(fields);
        listTemp.AddRange(whereFields);
        string[] allFields = listTemp.ToArray();
        listTemp.Clear();

        listTemp.AddRange(fieldValues);
        listTemp.AddRange(whereValues);
        string[] allValues = listTemp.ToArray();
        listTemp.Clear();

        string sql = null;
        OracleCommand command = null;
        try
        {
            addTableSchema(tableName);

            con.Open();
            sql = getUpdateString(tableName, fields, whereFields, whereValues);
            command = new OracleCommand(sql, con);
            List<OracleParameter> list = getUpdateParameters(tableName, allFields);
            for (int i = 0; i < list.Count; i++)
            {
                OracleParameter op = list[i];
                if (i < fields.Length)
                {
                    //要更新的欄位
                    op.Value = String.IsNullOrWhiteSpace(allValues[i]) ? 
                        (object)DBNull.Value : convertParameterValue(op, allValues[i]);
                    command.Parameters.Add(op);
                }
                else
                {
                    //where條件欄位
                    if (!String.IsNullOrWhiteSpace(allValues[i]))
                    {
                        op.Value = convertParameterValue(op, allValues[i]);
                        command.Parameters.Add(op);
                    }
                }
            }
            return command.ExecuteNonQuery();
        }
        catch (Exception ex)
        {
            writeLog(ex.Source + "\t" + ex.Message + "\r\n\t\t\t" + sql + "\r\n");
            throw ex;
        }
        finally
        {
            command.Dispose();
            con.Close();
        }
    }

    public int ExecuteInsert(string tableName, string[] fields, string[] fieldValues)
    {
        StringBuilder sb = new StringBuilder();
        bool isSbOk = toStringBuilder(sb, "INSERT", tableName, fields, fieldValues, null, null);
        writeLog(isSbOk.ToString() + "\r\n" + sb.ToString());


        string sql = null;
        OracleCommand command = null;
        try
        {
            addTableSchema(tableName);

            con.Open();
            sql = getInsertString(tableName, fields);
            command = new OracleCommand(sql, con);
            List<OracleParameter> list = getInsertParameters(tableName, fields);
            for (int i = 0; i < list.Count; i++)
            {
                OracleParameter op = list[i];
                op.Value = String.IsNullOrWhiteSpace(fieldValues[i]) ? 
                    (object)DBNull.Value : convertParameterValue(op, fieldValues[i]);
                command.Parameters.Add(op);
            }
            return command.ExecuteNonQuery();
        }
        catch (Exception ex)
        {
            writeLog(ex.Source + "\t" + ex.Message + "\r\n\t\t\t" + sql + "\r\n");
            throw ex;
        }
        finally
        {
            command.Dispose();
            con.Close();
        }
    }

    public int ExecuteDelete(string tableName, string[] whereFields, string[] whereValues)
    {
        StringBuilder sb = new StringBuilder();
        bool isSbOk = toStringBuilder(sb, "DELETE", tableName, null, null, whereFields, whereValues);
        writeLog(isSbOk.ToString() + "\r\n" + sb.ToString());


        string sql = null;
        OracleCommand command = null;
        try
        {
            addTableSchema(tableName);

            con.Open();
            sql = getDeleteString(tableName, whereFields, whereValues);
            command = new OracleCommand(sql, con);
            List<OracleParameter> list = getDeleteParameters(tableName, whereFields);
            for (int i = 0; i < list.Count; i++)
            {
                if (!String.IsNullOrWhiteSpace(whereValues[i]))
                {
                    OracleParameter op = list[i];
                    op.Value = convertParameterValue(op, whereValues[i]);
                    command.Parameters.Add(op);
                }
            }
            return command.ExecuteNonQuery();
        }
        catch (Exception ex)
        {
            writeLog(ex.Source + "\t" + ex.Message + "\r\n\t\t\t" + sql + "\r\n");
            throw ex;
        }
        finally
        {
            command.Dispose();
            con.Close();
        }
    }

    public DataSet ExecuteSelect(string tableName, string[] fields, string[] whereFields, string[] whereValues)
    {
        string sql = null;
        OracleCommand command = null;
        try
        {
            addTableSchema(tableName);

            con.Open();
            sql = getSelectString(tableName, fields, whereFields, whereValues);
            command = new OracleCommand(sql, con);
            List<OracleParameter> list = getSelectParameters(tableName, whereFields);
            for (int i = 0; i < list.Count; i++)
            {
                if (!String.IsNullOrWhiteSpace(whereValues[i]))
                {
                    OracleParameter op = list[i];
                    op.Value = convertParameterValue(op, whereValues[i]);
                    command.Parameters.Add(op);
                }
            }
            OracleDataAdapter da = new OracleDataAdapter(command);
            DataSet ds = new DataSet();
            da.Fill(ds, "0");
            return ds;
        }
        catch (Exception ex)
        {
            writeLog(ex.Source + "\t" + ex.Message + "\r\n\t\t\t" + sql + "\r\n");
            throw ex;
        }
        finally
        {
            command.Dispose();
            con.Close();
        }
    }

    public void ExecuteBatchUpdate(DbUpdate[] dbUpdate)
    {
        StringBuilder sb = new StringBuilder();
        bool isSbOk = toStringBuilder(sb, dbUpdate);
        writeLog(isSbOk.ToString() + "\r\n" + sb.ToString());

        OracleTransaction trans = null;
        string sql = null;
        int index = 0;
        try
        {
            con.Open();
            trans = con.BeginTransaction();
            List<string> listTemp = new List<string>();

            for (int i = 0; i < dbUpdate.Length; i++)
            {
                addTableSchema(dbUpdate[i].TableName);

                listTemp.AddRange(dbUpdate[i].Fields);
                listTemp.AddRange(dbUpdate[i].WhereFields);
                string[] allFields = listTemp.ToArray();
                listTemp.Clear();

                listTemp.AddRange(dbUpdate[i].FieldValues);
                listTemp.AddRange(dbUpdate[i].WhereValues);
                string[] allValues = listTemp.ToArray();
                listTemp.Clear();

                sql = getUpdateString(dbUpdate[i].TableName, dbUpdate[i].Fields, dbUpdate[i].WhereFields, dbUpdate[i].WhereValues);
                OracleCommand command = new OracleCommand(sql, con);
                try
                {
                    List<OracleParameter> list = getUpdateParameters(dbUpdate[i].TableName, allFields);
                    for (int j = 0; j < list.Count; j++)
                    {
                        OracleParameter op = list[j];
                        if (j < dbUpdate[i].Fields.Length)
                        {
                            //要更新的欄位
                            op.Value = String.IsNullOrWhiteSpace(allValues[j]) ?
                                (object)DBNull.Value : convertParameterValue(op, allValues[j]);
                            command.Parameters.Add(op);
                        }
                        else
                        {
                            //where條件欄位
                            if (!String.IsNullOrWhiteSpace(allValues[j]))
                            {
                                op.Value = convertParameterValue(op, allValues[j]);
                                command.Parameters.Add(op);
                            }
                        }
                    }
                    command.ExecuteNonQuery();
                }
                finally
                {
                    command.Dispose();
                }
                index++;
            }
            trans.Commit();
        }
        catch (Exception ex)
        {
            trans.Rollback();
            writeLog("Error Index:" + index.ToString() + "\r\n" + ex.Source + "\t" + 
                ex.Message + "\r\n\t\t\t" + sql + "\r\n");
            throw ex;
        }
        finally
        {
            con.Close();
        }
    }

    public void ExecuteBatchInsert(DbInsert[] dbInsert)
    {
        StringBuilder sb = new StringBuilder();
        bool isSbOk = toStringBuilder(sb, dbInsert);
        writeLog(isSbOk.ToString() + "\r\n" + sb.ToString());

        OracleTransaction trans = null;
        string sql = null;
        int index = 0;
        try
        {
            con.Open();
            trans = con.BeginTransaction();
            for (int i = 0; i < dbInsert.Length; i++)
            {
                addTableSchema(dbInsert[i].TableName);

                sql = getInsertString(dbInsert[i].TableName, dbInsert[i].Fields);
                OracleCommand command = new OracleCommand(sql, con);

                try
                {
                    List<OracleParameter> list = getInsertParameters(dbInsert[i].TableName, dbInsert[i].Fields);
                    for (int j = 0; j < list.Count; j++)
                    {
                        OracleParameter op = list[j];
                        op.Value = String.IsNullOrWhiteSpace(dbInsert[i].FieldValues[j]) ?
                            (object)DBNull.Value : convertParameterValue(op, dbInsert[i].FieldValues[j]);
                        command.Parameters.Add(op);
                     }
                    command.ExecuteNonQuery();
                }
                finally
                {
                    command.Dispose();
                }
                index++;
            }
            trans.Commit();
        }
        catch (Exception ex)
        {
            trans.Rollback();
            writeLog("Error Index:" + index.ToString() + "\r\n" + ex.Source + "\t" + 
                ex.Message + "\r\n\t\t\t" + sql + "\r\n");
            throw ex;
        }
        finally
        {
            con.Close();
        }
    }

    public void ExecuteBatchDelete(DbDelete[] dbDelete)
    {
        StringBuilder sb = new StringBuilder();
        bool isSbOk = toStringBuilder(sb, dbDelete);
        writeLog(isSbOk.ToString() + "\r\n" + sb.ToString());

        OracleTransaction trans = null;
        string sql = null;
        int index = 0;
        try
        {
            con.Open();
            trans = con.BeginTransaction();
            for (int i = 0; i < dbDelete.Length; i++)
            {
                addTableSchema(dbDelete[i].TableName);

                sql = getDeleteString(dbDelete[i].TableName, dbDelete[i].WhereFields, dbDelete[i].WhereValues);
                OracleCommand command = new OracleCommand(sql, con);

                try
                {
                    List<OracleParameter> list = getDeleteParameters(dbDelete[i].TableName, dbDelete[i].WhereFields);
                    for (int j = 0; j < list.Count; j++)
                    {
                        if (!String.IsNullOrWhiteSpace(dbDelete[i].WhereValues[j]))
                        {
                            OracleParameter op = list[j];
                            op.Value = convertParameterValue(op, dbDelete[i].WhereValues[j]);
                            command.Parameters.Add(op);
                        }
                    }
                    command.ExecuteNonQuery();
                }
                finally
                {
                    command.Dispose();
                }
                index++;
            }
            trans.Commit();
        }
        catch (Exception ex)
        {
            trans.Rollback();
            writeLog("Error Index:" + index.ToString() + "\r\n" + ex.Source + "\t" + 
                ex.Message + "\r\n\t\t\t" + sql + "\r\n");
            throw ex;
        }
        finally
        {
            con.Close();
        }
    }

    protected OracleCommand getInsertCommand(string tableName, string[] fields)
    {
        string[] insertFields = (string[])fields.Clone();
        for (int i = 0; i < insertFields.Length; i++)
        {
            int idx = insertFields[i].IndexOf('=');
            insertFields[i] = (idx == -1) ? insertFields[i] : insertFields[i].Substring(0, idx).Trim();
        }

        OracleCommand command = new OracleCommand();
        string sql = getInsertString(tableName, insertFields);
        List<OracleParameter> list = getInsertParameters(tableName, insertFields);
        command.Parameters.AddRange(list.ToArray());

        command.CommandText = sql;
        command.Prepare();
        return command;
    }

    protected OracleCommand getUpdateCommand(string tableName, string[] fields, string[] whereFields)
    {
        OracleCommand command = new OracleCommand();
        string sql = getUpdateString(tableName, fields, whereFields);

        List<OracleParameter> list = getUpdateParameters(tableName, fields);
        list.AddRange(getUpdateParameters(tableName, whereFields));
        command.Parameters.AddRange(list.ToArray());

        command.CommandText = sql;
        command.Prepare();
        return command;
    }

    protected OracleCommand getDeleteCommand(string tableName, string[] whereFields)
    {
        OracleCommand command = new OracleCommand();
        string sql = getDeleteString(tableName, whereFields);
        List<OracleParameter> list = getDeleteParameters(tableName, whereFields);
        command.Parameters.AddRange(list.ToArray());
        command.CommandText = sql;
        command.Prepare();
        return command;
    }

    protected void setInsertParameterValue(OracleCommand command, string[] fieldValues)
    {
        for (int i = 0; i < fieldValues.Length; i++)
        {
            OracleParameter op = command.Parameters[i];
            op.Value = String.IsNullOrWhiteSpace(fieldValues[i]) ?
                (object)DBNull.Value : convertParameterValue(op, fieldValues[i]);   
        }
    }

    protected void setUpdateParameterValue(OracleCommand command, string[] fieldValues, string[] whereValues)
    {
        for (int i = 0; i < fieldValues.Length; i++)
        {
            OracleParameter op = command.Parameters[i];
            op.Value = String.IsNullOrWhiteSpace(fieldValues[i]) ? 
                (object)DBNull.Value : convertParameterValue(op, fieldValues[i]);           
        }
        int count = 0;
        for (int i = 0; i < whereValues.Length; i++)
        {
            if (String.IsNullOrWhiteSpace(whereValues[i]))
            {
                command.Parameters.RemoveAt(count + fieldValues.Length);
            }
            else
            {
                OracleParameter op = command.Parameters[count + fieldValues.Length];
                op.Value = convertParameterValue(op, whereValues[i]);
                count++;
            }
        }
    }

    protected void setDeleteParameterValue(OracleCommand command, string[] whereValues)
    {
        int count = 0;
        for (int i = 0; i < whereValues.Length; i++)
        {
            if (String.IsNullOrWhiteSpace(whereValues[i]))
            {
                command.Parameters.RemoveAt(count);
            }
            else
            {
                OracleParameter op = command.Parameters[count];
                op.Value = convertParameterValue(op, whereValues[i]);
                count++;
            }
        }
    }

    public void ExecuteBatch(DbTreatment[] treatments)
    {
        StringBuilder sb = new StringBuilder();
        bool isSbOk = toStringBuilder(sb, treatments);
        writeLog(isSbOk.ToString() + "\r\n" + sb.ToString());

        OracleTransaction trans = null;
        OracleCommand commandI = null;
        OracleCommand commandU = null;
        OracleCommand commandD = null;
        int[] pkIdx = null;
        int[] nonPkIdx = null;
        int idx1 = 0;
        int idx2 = 0;
        string sql = null;

        try
        {
            con.Open();
            trans = con.BeginTransaction();
            for (int i = 0; i < treatments.Length; i++)
            {
                if (treatments[i].DMLType != DmlType.UserDefinition)
                {
                    addTableSchema(treatments[i].TableName);
                    addTablePK(treatments[i].TableName);
                }

                idx1 = i;
                switch (treatments[i].DMLType)
                {
                    case DmlType.UserDefinition:
                        commandD = new OracleCommand(treatments[i].CommandText);
                        commandD.Connection = con;
                        sql = commandD.CommandText;
                        break;
                    case DmlType.InsertNoConstraint:
                    case DmlType.Insert:
                        commandI = getInsertCommand(treatments[i].TableName, treatments[i].Fields);
                        commandI.Connection = con;
                        sql = commandI.CommandText;
                        break;
                    case DmlType.Update:
                        if (treatments[i].FieldValues.Length != treatments[i].WhereValues.Length)
                        {
                            throw new Exception("The FieldValues' length must equal to WhereValues");
                        }
                        commandU = getUpdateCommand(treatments[i].TableName, treatments[i].Fields, treatments[i].WhereFields);
                        commandU.Connection = con;
                        sql = commandU.CommandText;
                        break;
                    case DmlType.Delete:
                        commandD = getDeleteCommand(treatments[i].TableName, treatments[i].WhereFields);
                        commandD.Connection = con;
                        sql = commandD.CommandText;
                        break;
                    case DmlType.InsertUpdate:
                        commandI = getInsertCommand(treatments[i].TableName, treatments[i].Fields);
                        commandI.Connection = con;
                        sql = commandI.CommandText;

                        DataTable pkTable = pkTables[treatments[i].TableName];
                        string[] pks = new string[pkTable.Rows.Count];
                        for (int j = 0; j < pkTable.Rows.Count; j++)
                        {
                            pks[j] = pkTable.Rows[j]["COLUMN_NAME"].ToString();
                        }
                        pkIdx = new int[pks.Length];
                        nonPkIdx = new int[treatments[i].Fields.Length - pks.Length];
                        string[] nonPkFields = new string[treatments[i].Fields.Length - pks.Length];
                        int idx = 0;
                        for (int j = 0; j < treatments[i].Fields.Length; j++)
                        {
                            bool isFound = false;
                            for (int k = 0; k < pks.Length; k++)
                            {
                                if (treatments[i].Fields[j].ToUpper() == pks[k].ToUpper())
                                {
                                    pkIdx[k] = j;
                                    isFound = true;
                                    break;
                                }
                            }
                            if (isFound == false)
                            {
                                nonPkIdx[idx] = j;
                                nonPkFields[idx] = treatments[i].Fields[j];
                                idx++;
                            }
                        }

                        commandU = getUpdateCommand(treatments[i].TableName, nonPkFields, pks);
                        commandU.Connection = con;
                        sql = commandU.CommandText;
                        break;
                }


                switch (treatments[i].DMLType)
                {
                    case DmlType.UserDefinition:
                        try
                        {
                            commandD.ExecuteNonQuery();
                        }
                        finally
                        {
                            commandD.Dispose();
                        }
                        break;
                    case DmlType.InsertNoConstraint:
                    case DmlType.Insert:
                        try
                        {
                            for (int j = 0; j < treatments[i].FieldValues.Length; j++)
                            {
                                idx2 = j;
                                setInsertParameterValue(commandI, treatments[i].FieldValues[j]);
                                try
                                {
                                    commandI.ExecuteNonQuery();
                                }
                                catch (OracleException ex)
                                {
                                    if (treatments[i].DMLType == DmlType.Insert)
                                    {
                                        throw ex;
                                    }
                                }
                            }
                        }
                        finally
                        {
                            commandI.Dispose();
                        }
                        break;
                    case DmlType.Update:
                        try
                        {
                            OracleParameter[] ops = new OracleParameter[commandU.Parameters.Count];
                            commandU.Parameters.CopyTo(ops, 0);
                            for (int j = 0; j < treatments[i].FieldValues.Length; j++)
                            {
                                idx2 = j;
                                commandU.Parameters.Clear();
                                commandU.Parameters.AddRange(ops);
                                setUpdateParameterValue(commandU, treatments[i].FieldValues[j], treatments[i].WhereValues[j]);

                                commandU.CommandText = treatments[i].WhereValues[j].Any(x => String.IsNullOrWhiteSpace(x) == true) ?
                                    getUpdateString(treatments[i].TableName, treatments[i].Fields, treatments[i].WhereFields, treatments[i].WhereValues[j]) : sql;
                                commandU.ExecuteNonQuery();
                            }
                        }
                        finally
                        {
                            commandU.Dispose();
                        }
                        break;
                    case DmlType.Delete:
                        try
                        {
                            OracleParameter[] ops = new OracleParameter[commandD.Parameters.Count];
                            commandD.Parameters.CopyTo(ops, 0);
                            for (int j = 0; j < treatments[i].WhereValues.Length; j++)
                            {
                                idx2 = j;
                                commandD.Parameters.Clear();
                                commandD.Parameters.AddRange(ops);
                                setDeleteParameterValue(commandD, treatments[i].WhereValues[j]);

                                commandD.CommandText = treatments[i].WhereValues[j].Any(x => String.IsNullOrWhiteSpace(x) == true) ?
                                    getDeleteString(treatments[i].TableName, treatments[i].WhereFields, treatments[i].WhereValues[j]) : sql;
                                commandD.ExecuteNonQuery();
                            }
                        }
                        finally
                        {
                            commandD.Dispose();
                        }
                        break;

                    case DmlType.InsertUpdate:
                        try
                        {
                            for (int j = 0; j < treatments[i].FieldValues.Length; j++)
                            {
                                idx2 = j;
                                int affectRows = 0;
                                try
                                {
                                    if (pkIdx.Length > 0)   //避免因為沒有主鍵形成表的全部row內容被改為相同
                                    {
                                        string[] fieldValues = new string[nonPkIdx.Length];
                                        string[] whereValues = new string[pkIdx.Length];

                                        for (int k = 0; k < fieldValues.Length; k++)
                                        {
                                            fieldValues[k] = treatments[i].FieldValues[j][nonPkIdx[k]];
                                        }
                                        for (int k = 0; k < whereValues.Length; k++)
                                        {
                                            whereValues[k] = treatments[i].FieldValues[j][pkIdx[k]];
                                        }
                                        //insertupdate是根據pk欄位來更新,pk欄位不會有空值,故不須要針對where條件再處理
                                        setUpdateParameterValue(commandU, fieldValues, whereValues);
                                        affectRows = commandU.ExecuteNonQuery();
                                    }
                                    if (affectRows == 0)
                                    {
                                        setInsertParameterValue(commandI, treatments[i].FieldValues[j]);
                                        commandI.ExecuteNonQuery();
                                    }
                                }
                                catch (Exception oex)
                                {
                                    throw oex;
                                }
                            }
                        }
                        finally
                        {
                            commandI.Dispose();
                            commandU.Dispose();
                        }
                        break;
                }
            }
            trans.Commit();
        }
        catch (Exception ex)
        {
            trans.Rollback();
            writeLog("Error Index:" + idx1.ToString() + " / " + idx2.ToString() + "\r\n" + ex.Source + "\t" + 
                ex.Message + "\r\n\t\t\t" + sql + "\r\n");
            throw ex;
        }
        finally
        {
            con.Close();
        }
    }

    public void DealWithSingleTable(DbDML dbDML)
    {
        StringBuilder sb = new StringBuilder();
        bool isSbOk = toStringBuilder(sb, dbDML);
        writeLog(isSbOk.ToString() + "\r\n" + sb.ToString());

        OracleTransaction trans = null;
        string sql = null;

        List<string> lstPk = new List<string>();
        List<string> lstPkFields = new List<string>();
        List<string> lstNonPkFields = new List<string>();

        List<string> lstPkValues = new List<string>();
        List<string> lstNonPkValues = new List<string>();

        List<string> lstAllFields = new List<string>();
        List<string> lstAllValues = new List<string>();

        List<int> lstPkIdx = new List<int>();
        List<int> lstNonPkIdx = new List<int>();

        try
        {
            addTableSchema(dbDML.TableName);
            addTablePK(dbDML.TableName);

            foreach (DataRow row in pkTables[dbDML.TableName].Rows)
            {
                lstPk.Add(row["COLUMN_NAME"].ToString());
            }

            bool isPK = false;
            for (int i = 0; i < dbDML.Fields.Length; i++)
            {
                isPK = false;
                for (int j = 0; j < lstPk.Count; j++)
                {
                    if (dbDML.Fields[i].ToUpper() == lstPk[j].ToUpper())
                    {
                        isPK = true;
                        lstPkFields.Add(dbDML.Fields[i]);
                        lstPkIdx.Add(i);
                        break;
                    }
                }
                if (isPK == false)
                {
                    lstNonPkFields.Add(dbDML.Fields[i]);
                    lstNonPkIdx.Add(i);
                }
            }

            lstAllFields.AddRange(lstNonPkFields);
            lstAllFields.AddRange(lstPkFields);

            string sqlI = getInsertString(dbDML.TableName, dbDML.Fields);
            string sqlU = getUpdateString(dbDML.TableName, lstNonPkFields.ToArray(), lstPkFields.ToArray());

            con.Open();
            trans = con.BeginTransaction();
            OracleCommand command = null;

            List<OracleParameter> list = null;
            for (int i = 0; i < dbDML.FieldValues.Length; i++)
            {
                lstPkValues.Clear();
                lstNonPkValues.Clear();
                lstAllValues.Clear();

                foreach (int idx in lstPkIdx)
                {
                    lstPkValues.Add(dbDML.FieldValues[i][idx]);
                }

                foreach (int idx in lstNonPkIdx)
                {
                    lstNonPkValues.Add(dbDML.FieldValues[i][idx]);
                }

                lstAllValues.AddRange(lstNonPkValues);
                lstAllValues.AddRange(lstPkValues);

                try
                {
                    int affectRows = 0;
                    if (lstPkFields.Count > 0)
                    {
                        command = new OracleCommand(sqlU, con);
                        list = getUpdateParameters(dbDML.TableName, lstAllFields.ToArray());
                        for (int j = 0; j < list.Count; j++)
                        {
                            OracleParameter op = list[j];
                            op.Value = String.IsNullOrWhiteSpace(lstAllValues[j]) ? 
                                (object)DBNull.Value : convertParameterValue(op, lstAllValues[j]);
                            command.Parameters.Add(op);
                        }
                        affectRows = command.ExecuteNonQuery();
                    }
                    //更新rowcount==0 則採insert
                    if (affectRows == 0)
                    {
                        command = new OracleCommand(sqlI, con);
                        list = getInsertParameters(dbDML.TableName, dbDML.Fields);
                        for (int j = 0; j < list.Count; j++)
                        {
                            OracleParameter op = list[j];
                            op.Value = String.IsNullOrWhiteSpace(dbDML.FieldValues[i][j]) ?
                                (object)DBNull.Value : convertParameterValue(op, dbDML.FieldValues[i][j]);
                            command.Parameters.Add(op);
                        }
                        command.ExecuteNonQuery();
                    }
                }
                catch (Exception ex)
                {
                    throw ex;
                }
                finally
                {
                    if (command != null)
                    {
                        command.Dispose();
                    }
                }
            }

            trans.Commit();
        }
        catch (Exception ex)
        {
            trans.Rollback();
            writeLog(ex.Source + "\t" + ex.Message + "\r\n\t\t\t" + sql + "\r\n");
            throw ex;
        }
        finally
        {
            con.Close();
        }
    }

    private List<OracleParameter> getParameters(string tableName, string[] fields)
    {
        List<OracleParameter> list = new List<OracleParameter>();
        DataView dataView = schemaTables[tableName].DefaultView;

        bool isFieldMatched = false;
        OracleParameter param = null;

        for (int i = 0; i < fields.Length; i++)
        {
            isFieldMatched = false;
            for (int j = 0; j < dataView.Count; j++)
            {
                if (fields[i].ToUpper() == dataView[j].Row["COLUMN_NAME"].ToString().ToUpper())
                {
                    //若是資料型別為內建,例如 Integer,則沒有必要包含大小,但您也能夠指定預設大小。
                    isFieldMatched = true;
                    switch (dataView[j].Row["DATA_TYPE"].ToString())
                    {
                        case "DATE":
                            param = new OracleParameter(fields[i], OracleDbType.Date);
                            break;
                        case "TIMESTAMP":
                            param = new OracleParameter(fields[i], OracleDbType.TimeStamp);
                            break;
                        case "VARCHAR2":
                            param = new OracleParameter(fields[i], OracleDbType.Varchar2);
                            break;
                        case "NUMBER":
                            param = new OracleParameter(fields[i], OracleDbType.Decimal);
                            break;
                        case "XMLTYPE":
                            param = new OracleParameter(fields[i], OracleDbType.XmlType);
                            break;
                        case "BLOB":
                            param = new OracleParameter(fields[i], OracleDbType.Blob);
                            break;
                        default:
                            param = new OracleParameter();
                            break;
                    }
                    list.Add(param);
                    break;
                }
            }
            if (isFieldMatched == false)
            {
                list.Add(new OracleParameter());
            }
        }
        return list;
    }

    private List<OracleParameter> getInsertParameters(string tableName, string[] fields)
    {
        return getParameters(tableName, fields);
    }

    private List<OracleParameter> getDeleteParameters(string tableName, string[] whereFields)
    {
        return getParameters(tableName, whereFields);
    }

    private List<OracleParameter> getUpdateParameters(string tableName, string[] allFields)
    {
        return getParameters(tableName, allFields);
    }

    private List<OracleParameter> getSelectParameters(string tableName, string[] whereFields)
    {
        return getParameters(tableName, whereFields);
    }

    protected string getInsertString(string tableName, string[] fields)
    {
        StringBuilder sb = new StringBuilder();
        sb.Append(" INSERT INTO ");
        sb.Append(tableName);
        sb.Append(" (" + String.Join(",", fields) + ") ");
        sb.Append(" VALUES(");
        int count = 0;
        for (int i = 0; i < fields.Length; i++)
        {
            count++;
            sb.Append((i == 0 ? ":" : ",:") + count.ToString());
        }
        sb.Append(")");
        return sb.ToString();
    }

    protected string getUpdateString(string tableName, string[] fields, string[] whereFields)
    {
        return getUpdateString(tableName, fields, whereFields, null);
    }

    protected string getUpdateString(string tableName, string[] fields, string[] whereFields, string[] whereValues)
    {
        StringBuilder sb = new StringBuilder();
        sb.Append(" UPDATE ");
        sb.Append(tableName);
        sb.Append(" SET ");
        int count = 0;
        for (int i = 0; i < fields.Length; i++)
        {
            sb.Append(i == 0 ? "" : ",");
            count++;
            bool hasEqMark = fields[i].Contains('=');
            bool hasQuestionMark = fields[i].Contains('?') ;
            string temp = (hasEqMark ? ":" : "=:") + count.ToString();
            sb.Append(hasQuestionMark ? fields[i].Replace("?", temp) : fields[i] + temp);
        }

        if (whereFields.Length > 0)
        {
            for (int i = 0; i < whereFields.Length; i++)
            {
                sb.Append((i == 0 ? " WHERE " : " AND "));

                bool isNullValue = (whereValues == null ? false : String.IsNullOrWhiteSpace(whereValues[i]));
                if (!isNullValue)
                {
                    count++;
                }

                bool hasQuestionMark = whereFields[i].Contains('?') ;
                if(hasQuestionMark)
                {
                    sb.Append(whereFields[i].Replace("?", ":" + count.ToString()));
                }
                else
                {
                    bool hasOperator = hasSupportOperator(whereFields[i]);
                    sb.Append(whereFields[i] + (isNullValue ? " IS NULL " : (hasOperator ? ":" : "=:") + count.ToString()));
                }
            }
        }
        return sb.ToString();
    }

    protected string getDeleteString(string tableName, string[] whereFields)
    {
        return getDeleteString(tableName, whereFields, null);
    }

    protected string getDeleteString(string tableName, string[] whereFields, string[] whereValues)
    {
        StringBuilder sb = new StringBuilder();
        sb.Append(" DELETE ");
        sb.Append(tableName);

        int count = 0;
        if (whereFields.Length > 0)
        {
            for (int i = 0; i < whereFields.Length; i++)
            {
                sb.Append((i == 0 ? " WHERE " : " AND "));

                bool isNullValue = (whereValues == null ? false : String.IsNullOrWhiteSpace(whereValues[i]));
                if (!isNullValue)
                {
                    count++;
                }

                bool hasQuestionMark = whereFields[i].Contains('?');
                if (hasQuestionMark)
                {
                    sb.Append(whereFields[i].Replace("?", ":" + count.ToString()));
                }
                else
                {
                    bool hasOperator = hasSupportOperator(whereFields[i]);
                    sb.Append(whereFields[i] + (isNullValue ? " IS NULL " : (hasOperator ? ":" : "=:") + count.ToString()));
                }
            }
        }
        return sb.ToString();
    }

    protected string getSelectString(string tableName, string[] fields, string[] whereFields)
    {
        return getSelectString(tableName, fields, whereFields, null);
    }

    protected string getSelectString(string tableName, string[] fields, string[] whereFields, string[] whereValues)
    {
        StringBuilder sb = new StringBuilder();
        sb.Append(" SELECT " + String.Join(",", fields) + " FROM " + tableName);

        int count = 0;
        if (whereFields.Length > 0)
        {
            for (int i = 0; i < whereFields.Length; i++)
            {
                sb.Append((i == 0 ? " WHERE " : " AND "));

                bool isNullValue = (whereValues == null ? false : String.IsNullOrWhiteSpace(whereValues[i]));
                if (!isNullValue)
                {
                    count++;
                }

                bool hasQuestionMark = whereFields[i].Contains('?');
                if (hasQuestionMark)
                {
                    sb.Append(whereFields[i].Replace("?", ":" + count.ToString()));
                }
                else
                {
                    bool hasOperator = hasSupportOperator(whereFields[i]);
                    sb.Append(whereFields[i] + (isNullValue ? " IS NULL " : (hasOperator ? ":" : "=:") + count.ToString()));
                }
            }
        }
        return sb.ToString();
    }

    private bool hasSupportOperator(string target)
    {
        string[] operators = new string[] { ">", "<", "LIKE" };
        bool isFound = false;
        foreach (string op in operators)
        {
            if (target.IndexOf(op, StringComparison.CurrentCultureIgnoreCase) >= 0)
            {
                isFound = true;
                break;
            }
        }
        return isFound;
    }

    //"若是要傳回全部table的table schema,請傳入null")]
    public DataSet GetTableSchema(string tableName)
    {
        string schema = null;
        string dblink = null;

        if (String.IsNullOrEmpty(tableName) == false)
        {
            tableName = tableName.ToUpper();
            int idx = tableName.IndexOf('@');
            if (idx >= 0)
            {
                tableName = tableName.Substring(0, idx);
                dblink = tableName.Substring(idx);
            }
            idx = tableName.IndexOf('.');
            if (idx >= 0)
            {
                schema = tableName.Substring(0, idx);
                tableName = tableName.Substring(idx + 1);
            }
        }

        try
        {
            string sql = "SELECT TABLE_NAME, COLUMN_NAME, DATA_TYPE, DATA_LENGTH, DATA_PRECISION, NULLABLE, COLUMN_ID ";
            if (String.IsNullOrEmpty(schema))
            {
                sql += " FROM USER_TAB_COLUMNS" + (dblink == null ? "" : dblink);
                if (String.IsNullOrEmpty(tableName) == false)
                {
                    sql += " WHERE TABLE_NAME='" + tableName + "'";
                }
            }
            else
            {
                sql += " FROM DBA_TAB_COLUMNS" + (dblink == null ? "" : dblink);
                sql += " WHERE OWNER='" + schema + "' ";
                if (String.IsNullOrEmpty(tableName) == false)
                {
                    sql += " AND TABLE_NAME='" + tableName + "'";
                }
            }

            sql += " ORDER BY TABLE_NAME, COLUMN_ID ";

            return ExecuteQuery(sql);
        }
        catch (Exception ex)
        {
            writeLog(ex.Source + "\t" + ex.Message + "\r\n");
            throw ex;
        }
    }

    public DataSet GetTablePK(string tableName)
    {
        string schema = null;
        string dblink = null;

        if (String.IsNullOrEmpty(tableName) == false)
        {
            tableName = tableName.ToUpper();
            int idx = tableName.IndexOf('@');
            if (idx >= 0)
            {
                tableName = tableName.Substring(0, idx);
                dblink = tableName.Substring(idx);
            }
            idx = tableName.IndexOf('.');
            if (idx >= 0)
            {
                schema = tableName.Substring(0, idx);
                tableName = tableName.Substring(idx + 1);
            }
        }

        try
        {
            string sql = "SELECT B.COLUMN_NAME, B.POSITION, A.TABLE_NAME ";
            if (String.IsNullOrEmpty(schema))
            {
                sql += " FROM USER_CONSTRAINTS" + (dblink == null ? "" : dblink) + " A, " +
                    " USER_CONS_COLUMNS" + (dblink == null ? "" : dblink) + " B ";
            }
            else
            {
                sql += " FROM DBA_CONSTRAINTS" + (dblink == null ? "" : dblink) + " A, " +
                    " DBA_CONS_COLUMNS" + (dblink == null ? "" : dblink) + " B ";
            }
            sql += " WHERE A.OWNER=B.OWNER AND A.CONSTRAINT_NAME=B.CONSTRAINT_NAME ";
            if (String.IsNullOrEmpty(schema) == false)
            {
                sql += " AND A.OWNER='" + schema + "' AND B.OWNER='" + schema + "'";
            }
            if (String.IsNullOrEmpty(tableName) == false)
            {
                sql += " AND A.TABLE_NAME='" + tableName + "'";
            }
            sql += " AND A.CONSTRAINT_TYPE='P' ORDER BY POSITION";

            return ExecuteQuery(sql);
        }
        catch (Exception ex)
        {
            writeLog(ex.Source + "\t" + ex.Message + "\r\n");
            throw ex;
        }
    }

    protected void addTableSchema(string tableName)
    {
        if (schemaTables == null)
        {
            schemaTables = new Dictionary<string, DataTable>();
        }

        if (!schemaTables.ContainsKey(tableName))
        {
            schemaTables[tableName] = this.GetTableSchema(tableName).Tables[0];
        }
    }

    protected void addTablePK(string tableName)
    {
        if (pkTables == null)
        {
            pkTables = new Dictionary<string, DataTable>();
        }

        if (!pkTables.ContainsKey(tableName))
        {
            pkTables[tableName] = this.GetTablePK(tableName).Tables[0];
        }
    }

    public void SendMail(MailInformation mail)
    {
        string smtpHost = "nhsmtp.cminl.oa";
        MailAddress ma = null;
        try
        {
            System.Net.Mail.MailMessage message = new System.Net.Mail.MailMessage();
            message.Subject = HttpUtility.UrlDecode(mail.Subject);
            message.IsBodyHtml = mail.IsBodyHtml;
            message.Body = HttpUtility.UrlDecode(mail.Body) + (mail.IsBodyHtml ? "<br><br><br>" : "\r\n\r\n");
            message.SubjectEncoding = Encoding.GetEncoding(mail.CodePage);
            message.BodyEncoding = Encoding.GetEncoding(mail.CodePage);

            ma = new MailAddress(mail.From.Address, mail.From.Name);
            message.From = ma;

            if (mail.Object.ToAddresses != null)
            {
                foreach (string address in mail.Object.ToAddresses)
                {
                    if (address.Contains("@"))
                    {
                        message.To.Add(address.Trim());
                    }
                }
            }

            if (mail.Object.CcAddresses != null)
            {
                foreach (string address in mail.Object.CcAddresses)
                {
                    if (address.Contains("@"))
                    {
                        message.CC.Add(address.Trim());
                    }
                }
            }

            if (mail.Object.BccAddresses != null)
            {
                foreach (string address in mail.Object.BccAddresses)
                {
                    if (address.Contains("@"))
                    {
                        message.Bcc.Add(address.Trim());
                    }
                }
            }

            if (mail.AttachFiles != null)
            {
                foreach (AttachFile file in mail.AttachFiles)
                {
                    byte[] bytes = Convert.FromBase64String(file.Base64Content);
                    MemoryStream ms = new MemoryStream(bytes);
                    Attachment attach = new Attachment(ms, file.FileName, getMediaType(file.FileName));
                    attach.NameEncoding = Encoding.GetEncoding(mail.CodePage);
                    ContentDisposition disposition = attach.ContentDisposition;
                    disposition.Inline = (file.Inline ? true : false);
                    message.Attachments.Add(attach);
                }
            }

            System.Net.Mail.SmtpClient smtp = new System.Net.Mail.SmtpClient(smtpHost);
            smtp.Send(message);
        }
        catch (Exception ex)
        {
            writeLog("Unable to send mail.\r\n\t" + ex.GetBaseException().Message + "\r\n");
        }
    }

    private string getMediaType(string fileName)
    {
        string ext = fileName.Substring(fileName.LastIndexOf('.') + 1);
        string mediaType = MediaTypeNames.Application.Octet;
        switch (ext.ToLower())
        {
            case "pdf":
                mediaType = MediaTypeNames.Application.Pdf;
                break;
            case "rtf":
                mediaType = MediaTypeNames.Application.Rtf;
                break;
            case "zip":
                mediaType = MediaTypeNames.Application.Zip;
                break;
            case "gif":
                mediaType = MediaTypeNames.Image.Gif;
                break;
            case "jpg":
            case "jpeg":
                mediaType = MediaTypeNames.Image.Jpeg;
                break;
            case "tif":
            case "tiff":
                mediaType = MediaTypeNames.Image.Tiff;
                break;
            case "htm":
            case "html":
                mediaType = MediaTypeNames.Text.Html;
                break;
            case "xml":
                mediaType = MediaTypeNames.Text.Xml;
                break;
            case "txt":
                mediaType = MediaTypeNames.Text.Plain;
                break;
        }
        return mediaType;
    }

    protected void writeLog(string detailDesc)
    {
        string fullText = null;

        using (var mutex = new Mutex(false, this.GetType().Name))
        {
            if (!mutex.WaitOne(TimeSpan.FromSeconds(3), false))
            {
                return;
            }

            try
            {
                string logFile = HostingEnvironment.MapPath("~/Log/" + this.GetType().Name + "_" + DateTime.Now.ToString("yyyyMMdd") + ".txt");
                fullText = DateTime.Now.ToString("yyyy/MM/dd HH:mm:ss") + "\t" + clientAddress + "\t" + detailDesc;
                checkFile(logFile);
                using (System.IO.StreamWriter sw = System.IO.File.AppendText(logFile))
                {
                    sw.WriteLine(fullText);
                    sw.Flush();
                    sw.Close();
                    sw.Dispose();
                }
            }
            catch
            { }
            finally
            {
                try
                {
                    mutex.ReleaseMutex();
                }
                catch
                {
                }
            }
        }
    }

    private void checkFile(string fileName)
    {
        if (!System.IO.File.Exists(fileName))
        {
            System.IO.StreamWriter sw = System.IO.File.CreateText(fileName);
            sw.Close();
            sw.Dispose();
        }
    }

    protected bool toStringBuilder(StringBuilder sb, DbDML dbDML)
    {
        try
        {
            sb.Append("DbDML \t TableName: " + dbDML.TableName + "\r\n");
            sb.Append(String.Join("^", dbDML.Fields) + "\r\n");
            for (int i = 0; i < dbDML.FieldValues.GetLength(0); i++)
            {
                sb.Append(String.Join("^", dbDML.FieldValues[i]) + "\r\n");
            }
            return true;
        }
        catch (Exception ex)
        {
            writeLog(ex.Message + "\r\n" + ex.TargetSite.Name + "\r\n");
            return false;
        }
    }

    protected bool toStringBuilder(StringBuilder sb, DbTreatment[] dbTreatments)
    {
        try
        {
            foreach (DbTreatment dbTreatment in dbTreatments)
            {
                sb.Append("DbTreatment \t TableName: " + dbTreatment.TableName + "\t DML:" + dbTreatment.DMLType.ToString() + "\r\n");
                sb.Append("CommandText: \t" + (String.IsNullOrEmpty(dbTreatment.CommandText) ? "" : dbTreatment.CommandText) + "\r\n");
                if (dbTreatment.Fields != null && dbTreatment.Fields.Length > 0)
                {
                    sb.Append(String.Join("^", dbTreatment.Fields) + "\r\n");
                    for (int i = 0; i < dbTreatment.FieldValues.GetLength(0); i++)
                    {
                        sb.Append(String.Join("^", dbTreatment.FieldValues[i]) + "\r\n");
                    }
                }
                if (dbTreatment.WhereFields != null && dbTreatment.WhereFields.Length > 0)
                {
                    sb.Append(String.Join("^", dbTreatment.WhereFields) + "\r\n");
                    for (int i = 0; i < dbTreatment.WhereValues.GetLength(0); i++)
                    {
                        sb.Append(String.Join("^", dbTreatment.WhereValues[i]) + "\r\n");
                    }
                }
            }
            return true;
        }
        catch (Exception ex)
        {
            writeLog(ex.Message + "\r\n" + ex.TargetSite.Name + "\r\n");
            return false;
        }
    }

    protected bool toStringBuilder(StringBuilder sb, DbInsert[] dbInserts)
    {
        try
        {
            foreach (DbInsert dbInsert in dbInserts)
            {
                sb.Append("DbInsert \t TableName: " + dbInsert.TableName + "\r\n");
                sb.Append(String.Join("^", dbInsert.Fields) + "\r\n");
                sb.Append(String.Join("^", dbInsert.FieldValues) + "\r\n");
            }
            return true;
        }
        catch (Exception ex)
        {
            writeLog(ex.Message + "\r\n" + ex.TargetSite.Name + "\r\n");
            return false;
        }
    }

    protected bool toStringBuilder(StringBuilder sb, DbUpdate[] dbUpdates)
    {
        try
        {
            foreach (DbUpdate dbUpdate in dbUpdates)
            {
                sb.Append("DbUpdate \t TableName: " + dbUpdate.TableName + "\r\n");
                sb.Append(String.Join("^", dbUpdate.Fields) + "\r\n");
                sb.Append(String.Join("^", dbUpdate.FieldValues) + "\r\n");
                sb.Append(String.Join("^", dbUpdate.WhereFields) + "\r\n");
                sb.Append(String.Join("^", dbUpdate.WhereValues) + "\r\n");
            }
            return true;
        }
        catch (Exception ex)
        {
            writeLog(ex.Message + "\r\n" + ex.TargetSite.Name + "\r\n");
            return false;
        }
    }

    protected bool toStringBuilder(StringBuilder sb, DbDelete[] dbDeletes)
    {
        try
        {
            foreach (DbDelete dbDelete in dbDeletes)
            {
                sb.Append("DbDelete \t TableName: " + dbDelete.TableName + "\r\n");
                sb.Append(String.Join("^", dbDelete.WhereFields) + "\r\n");
                sb.Append(String.Join("^", dbDelete.WhereValues) + "\r\n");
            }
            return true;
        }
        catch (Exception ex)
        {
            writeLog(ex.Message + "\r\n" + ex.TargetSite.Name + "\r\n");
            return false;
        }
    }

    protected bool toStringBuilder(StringBuilder sb, string dmlType, string tableName, string[] fields, string[] fieldValues, string[] whereFields, string[] whereValues)
    {
        try
        {
            sb.Append("DML Type: " + dmlType + "\t TableName: " + tableName + "\r\n");
            if (fields != null && fields.Length > 0)
            {
                sb.Append(String.Join("^", fields) + "\r\n");
                sb.Append(String.Join("^", fieldValues) + "\r\n");
            }

            if (whereFields != null && whereFields.Length > 0)
            {
                sb.Append(String.Join("^", whereFields) + "\r\n");
                sb.Append(String.Join("^", whereValues) + "\r\n");
            }
            return true;
        }
        catch (Exception ex)
        {
            writeLog(ex.Message + "\r\n" + ex.TargetSite.Name + "\r\n");
            return false;
        }
    }

    private string clientAddress
    {
        get
        {
            RemoteEndpointMessageProperty clientEndpoint =
                OperationContext.Current.IncomingMessageProperties[RemoteEndpointMessageProperty.Name] as RemoteEndpointMessageProperty;
            return clientEndpoint.Address;
        }
    }



    
}
相關文章
相關標籤/搜索