326 lines
11 KiB
C#
Raw Normal View History

using System;
using System.Data;
using System.Data.SqlClient;
using System.Data.SqlTypes;
using System.Collections;
using System.Collections.Generic;
using System.Text;
using System.Xml;
using System.Reflection;
using System.IO;
using System.Runtime.Serialization.Formatters.Binary;
using System.ComponentModel.Design.Serialization;
using System.Workflow.Runtime;
using System.Workflow.ComponentModel;
using System.Workflow.ComponentModel.Serialization;
using System.Workflow.Runtime.Hosting;
namespace System.Workflow.Runtime.Tracking
{
[Obsolete("The System.Workflow.* types are deprecated. Instead, please use the new types from System.Activities.*")]
public sealed class SqlTrackingQuery
{
string _connectionString = null;
public SqlTrackingQuery()
{
}
public SqlTrackingQuery(string connectionString)
{
if (null == connectionString)
throw new ArgumentNullException("connectionString");
_connectionString = connectionString;
}
public string ConnectionString
{
get { return _connectionString; }
set
{
if (null == value)
throw new ArgumentNullException("value");
_connectionString = value;
}
}
public bool TryGetWorkflow(Guid workflowInstanceId, out SqlTrackingWorkflowInstance workflowInstance)
{
SqlCommand cmd = BuildCommand(workflowInstanceId);
SqlDataReader reader = null;
workflowInstance = null;
try
{
cmd.Connection = GetConnection();
reader = cmd.ExecuteReader(CommandBehavior.CloseConnection);
//
// There will only be 1 row
if (reader.Read())
{
workflowInstance = BuildInstance(reader);
return true;
}
else
return false;
}
finally
{
if (null != reader)
reader.Close();
if (null != cmd && null != cmd.Connection && ConnectionState.Closed != cmd.Connection.State)
cmd.Connection.Close();
}
}
public IList<SqlTrackingWorkflowInstance> GetWorkflows(SqlTrackingQueryOptions options)
{
if (null == options)
throw new ArgumentNullException("options");
if (null != options.TrackingDataItems)
{
foreach (TrackingDataItemValue val in options.TrackingDataItems)
{
if (null == val.QualifiedName)
throw new ArgumentNullException("options.TrackingDataItems.QualifiedName");
if (null == val.FieldName)
throw new ArgumentNullException("options.TrackingDataItems.FieldName");
}
}
SqlCommand cmd = BuildCommand(options);
SqlDataReader reader = null;
List<SqlTrackingWorkflowInstance> inst = new List<SqlTrackingWorkflowInstance>();
try
{
cmd.Connection = GetConnection();
reader = cmd.ExecuteReader(CommandBehavior.CloseConnection);
//
// There will only be 1 row
while (reader.Read())
{
inst.Add(BuildInstance(reader));
}
}
finally
{
if (null != reader)
reader.Close();
if (null != cmd && null != cmd.Connection && ConnectionState.Closed != cmd.Connection.State)
cmd.Connection.Close();
}
return inst;
}
private SqlTrackingWorkflowInstance BuildInstance(SqlDataReader reader)
{
return SqlTrackingQuery.BuildInstance(reader, _connectionString);
}
internal static SqlTrackingWorkflowInstance BuildInstance(SqlDataReader reader, string connectionString)
{
if (null == reader)
throw new ArgumentNullException("reader");
if (reader.IsClosed)
throw new ArgumentException(ExecutionStringManager.InvalidSqlDataReader, "reader");
SqlTrackingWorkflowInstance inst = new SqlTrackingWorkflowInstance(connectionString);
inst.WorkflowInstanceId = reader.GetGuid(1);
inst.WorkflowInstanceInternalId = reader.GetInt64(2);
inst.Initialized = reader.GetDateTime(3);
if (DBNull.Value == reader[4])
inst.InvokingWorkflowInstanceId = Guid.Empty;
else
inst.InvokingWorkflowInstanceId = reader.GetGuid(4);
inst.Status = (WorkflowStatus)reader.GetInt32(5);
//
// Xaml only workflows do not have types
if (!reader.IsDBNull(6))
{
string fullName = reader.GetString(6), assemblyName = reader.GetString(7);
inst.WorkflowType = Type.GetType(fullName + ", " + assemblyName, true, false);
}
return inst;
}
private SqlConnection GetConnection()
{
if (null == _connectionString)
{
throw new InvalidOperationException(ExecutionStringManager.MissingConnectionString);
}
SqlConnection conn = new SqlConnection(_connectionString);
conn.Open();
return conn;
}
private SqlCommand BuildCommand(Guid workflowInstanceId)
{
SqlCommand cmd = new SqlCommand("[dbo].[GetWorkflows]");
cmd.CommandType = CommandType.StoredProcedure;
SqlParameter param = new SqlParameter();
param.ParameterName = "@WorkflowInstanceId";
param.SqlDbType = SqlDbType.UniqueIdentifier;
param.Value = workflowInstanceId;
cmd.Parameters.Add(param);
return cmd;
}
private SqlCommand BuildCommand(SqlTrackingQueryOptions opt)
{
SqlCommand cmd = new SqlCommand("[dbo].[GetWorkflows]");
cmd.CommandType = CommandType.StoredProcedure;
SqlParameter param = new SqlParameter();
if (opt.WorkflowStatus.HasValue)
{
param.ParameterName = "@WorkflowStatusId";
param.SqlDbType = SqlDbType.TinyInt;
param.Value = opt.WorkflowStatus.Value;
cmd.Parameters.Add(param);
//
// If one of the range values is set we have a date range constraint
if (DateTime.MinValue != opt.StatusMinDateTime || DateTime.MaxValue != opt.StatusMaxDateTime)
{
param = new SqlParameter();
param.ParameterName = "@StatusMinDateTime";
param.SqlDbType = SqlDbType.DateTime;
if (opt.StatusMinDateTime < SqlDateTime.MinValue.Value)
param.Value = SqlDateTime.MinValue.Value;
else
param.Value = opt.StatusMinDateTime;
cmd.Parameters.Add(param);
param = new SqlParameter();
param.ParameterName = "@StatusMaxDateTime";
param.SqlDbType = SqlDbType.DateTime;
if (opt.StatusMaxDateTime > SqlDateTime.MaxValue.Value)
param.Value = SqlDateTime.MaxValue.Value;
else
param.Value = opt.StatusMaxDateTime;
cmd.Parameters.Add(param);
}
}
if (null != opt.WorkflowType)
{
param = new SqlParameter("@TypeFullName", opt.WorkflowType.FullName);
param.SqlDbType = SqlDbType.NVarChar;
param.Size = 128;
cmd.Parameters.Add(param);
param = new SqlParameter("@AssemblyFullName", opt.WorkflowType.Assembly.FullName);
param.SqlDbType = SqlDbType.NVarChar;
param.Size = 128;
cmd.Parameters.Add(param);
}
if (null != opt.TrackingDataItems && opt.TrackingDataItems.Count > 0)
BuildArtifactParameters(cmd, opt.TrackingDataItems);
return cmd;
}
private void BuildArtifactParameters(SqlCommand cmd, IList<TrackingDataItemValue> artifacts)
{
if (null == artifacts || 0 == artifacts.Count)
return;
StringBuilder sb = new StringBuilder();
XmlWriter writer = XmlWriter.Create(sb);
try
{
writer.WriteStartDocument();
writer.WriteStartElement("TrackingDataItems");
foreach (TrackingDataItemValue art in artifacts)
{
writer.WriteStartElement("TrackingDataItem");
writer.WriteElementString("QualifiedName", art.QualifiedName);
writer.WriteElementString("FieldName", art.FieldName);
//
// If data value is null don't write the node as
// the proc sees no DataValue node as null and matches null rows.
// This allows us to match null, "", and positive length strings
if (null != art.DataValue)
writer.WriteElementString("DataValue", art.DataValue);
writer.WriteEndElement();
}
writer.WriteEndElement();
writer.WriteEndDocument();
}
finally
{
writer.Flush();
writer.Close();
}
SqlParameter param = new SqlParameter();
param.ParameterName = "@TrackingDataItems";
param.SqlDbType = SqlDbType.NText;
param.Value = sb.ToString();
cmd.Parameters.Add(param);
}
}
[Obsolete("The System.Workflow.* types are deprecated. Instead, please use the new types from System.Activities.*")]
public class SqlTrackingQueryOptions
{
private DateTime _min = DateTime.MinValue, _max = DateTime.MaxValue;
private WorkflowStatus? _status = new Nullable<WorkflowStatus>();
private Type _type = null;
private List<TrackingDataItemValue> _dataItems = new List<TrackingDataItemValue>();
public Type WorkflowType
{
get { return _type; }
set { _type = value; }
}
public WorkflowStatus? WorkflowStatus
{
get { return _status; }
set { _status = value; }
}
public DateTime StatusMinDateTime
{
get { return _min; }
set { _min = value; }
}
public DateTime StatusMaxDateTime
{
get { return _max; }
set { _max = value; }
}
public IList<TrackingDataItemValue> TrackingDataItems
{
get { return _dataItems; }
}
public void Clear()
{
_min = DateTime.MinValue;
_max = DateTime.MaxValue;
_status = new Nullable<WorkflowStatus>();
_type = null;
_dataItems = new List<TrackingDataItemValue>();
}
}
}