| | | 1 | | using System.Diagnostics.CodeAnalysis; |
| | | 2 | | using Elsa.Workflows.Management.Entities; |
| | | 3 | | using Elsa.Workflows.Management.Enums; |
| | | 4 | | using Elsa.Workflows.Management.Models; |
| | | 5 | | using System.Linq.Dynamic.Core; |
| | | 6 | | |
| | | 7 | | namespace Elsa.Workflows.Management.Filters; |
| | | 8 | | |
| | | 9 | | /// <summary> |
| | | 10 | | /// A filter for querying workflow instances. |
| | | 11 | | /// </summary> |
| | | 12 | | public class WorkflowInstanceFilter |
| | | 13 | | { |
| | 1 | 14 | | private static readonly string[] TimestampFilterColumns = |
| | 1 | 15 | | [ |
| | 1 | 16 | | nameof(WorkflowInstance.CreatedAt), |
| | 1 | 17 | | nameof(WorkflowInstance.UpdatedAt), |
| | 1 | 18 | | nameof(WorkflowInstance.FinishedAt) |
| | 1 | 19 | | ]; |
| | | 20 | | |
| | | 21 | | /// <summary> |
| | | 22 | | /// The workflow instance timestamp columns that can be used by <see cref="TimestampFilters"/>. |
| | | 23 | | /// </summary> |
| | 2 | 24 | | public static IReadOnlyCollection<string> AllowedTimestampFilterColumns { get; } = Array.AsReadOnly(TimestampFilterC |
| | | 25 | | |
| | | 26 | | /// <summary> |
| | | 27 | | /// Filter workflow instances by ID. |
| | | 28 | | /// </summary> |
| | 477 | 29 | | public string? Id { get; set; } |
| | | 30 | | |
| | | 31 | | /// <summary> |
| | | 32 | | /// Filter workflow instances by IDs. |
| | | 33 | | /// </summary> |
| | 262 | 34 | | public ICollection<string>? Ids { get; set; } |
| | | 35 | | |
| | | 36 | | /// <summary> |
| | | 37 | | /// Filter workflow instances that match the specified search term. |
| | | 38 | | /// </summary> |
| | 240 | 39 | | public string? SearchTerm { get; set; } |
| | | 40 | | |
| | | 41 | | /// <summary> |
| | | 42 | | /// Filter workflow instances that match the specified name. |
| | | 43 | | /// </summary> |
| | 242 | 44 | | public string? Name { get; set; } |
| | | 45 | | |
| | | 46 | | /// <summary> |
| | | 47 | | /// Filter workflow instances by definition ID. |
| | | 48 | | /// </summary> |
| | 285 | 49 | | public string? DefinitionId { get; set; } |
| | | 50 | | |
| | | 51 | | /// <summary> |
| | | 52 | | /// Filter workflow instances by definition version ID. |
| | | 53 | | /// </summary> |
| | 242 | 54 | | public string? DefinitionVersionId { get; set; } |
| | | 55 | | |
| | | 56 | | /// <summary> |
| | | 57 | | /// Filter workflow instances by definition IDs. |
| | | 58 | | /// </summary> |
| | 254 | 59 | | public ICollection<string>? DefinitionIds { get; set; } |
| | | 60 | | |
| | | 61 | | /// <summary> |
| | | 62 | | /// Filter workflow instances by definition version IDs. |
| | | 63 | | /// </summary> |
| | 242 | 64 | | public ICollection<string>? DefinitionVersionIds { get; set; } |
| | | 65 | | |
| | | 66 | | /// <summary> |
| | | 67 | | /// Filter workflow instances by version. |
| | | 68 | | /// </summary> |
| | 242 | 69 | | public int? Version { get; set; } |
| | | 70 | | |
| | | 71 | | /// <summary> |
| | | 72 | | /// Filter workflow instances by their parent instance IDs. |
| | | 73 | | /// </summary> |
| | 253 | 74 | | public ICollection<string>? ParentWorkflowInstanceIds { get; set; } |
| | | 75 | | |
| | | 76 | | /// <summary> |
| | | 77 | | /// Filter workflow instances by correlation ID. |
| | | 78 | | /// </summary> |
| | 248 | 79 | | public string? CorrelationId { get; set; } |
| | | 80 | | |
| | | 81 | | /// <summary> |
| | | 82 | | /// Filter workflow instances by correlation IDs. |
| | | 83 | | /// </summary> |
| | 242 | 84 | | public ICollection<string>? CorrelationIds { get; set; } |
| | | 85 | | |
| | | 86 | | /// <summary> |
| | | 87 | | /// Filter workflow instances by status. |
| | | 88 | | /// </summary> |
| | 480 | 89 | | public WorkflowStatus? WorkflowStatus { get; set; } |
| | | 90 | | |
| | | 91 | | /// <summary> |
| | | 92 | | /// Filter workflow instances by a set of statuses. |
| | | 93 | | /// </summary> |
| | 242 | 94 | | public ICollection<WorkflowStatus>? WorkflowStatuses { get; set; } |
| | | 95 | | |
| | | 96 | | /// <summary> |
| | | 97 | | /// Filter workflow instances by sub-status. |
| | | 98 | | /// </summary> |
| | 280 | 99 | | public WorkflowSubStatus? WorkflowSubStatus { get; set; } |
| | | 100 | | |
| | | 101 | | /// <summary> |
| | | 102 | | /// Filter workflow instances by a set of sub-status. |
| | | 103 | | /// </summary> |
| | 242 | 104 | | public ICollection<WorkflowSubStatus>? WorkflowSubStatuses { get; set; } |
| | | 105 | | |
| | | 106 | | /// <summary> |
| | | 107 | | /// Filter workflow instances by whether they are executing. |
| | | 108 | | /// </summary> |
| | 458 | 109 | | public bool? IsExecuting { get; set; } |
| | | 110 | | |
| | | 111 | | /// <summary> |
| | | 112 | | /// Filter workflow instances by whether they have incidents. |
| | | 113 | | /// </summary> |
| | 242 | 114 | | public bool? HasIncidents { get; set; } |
| | | 115 | | |
| | | 116 | | /// <summary> |
| | | 117 | | /// Filter on workflows that are system workflows. |
| | | 118 | | /// </summary> |
| | 242 | 119 | | public bool? IsSystem { get; set; } |
| | | 120 | | |
| | | 121 | | /// <summary> |
| | | 122 | | /// Filter workflow instances that are older than the specified timestamp. |
| | | 123 | | /// </summary> |
| | 458 | 124 | | public DateTimeOffset? BeforeLastUpdated { get; set; } |
| | | 125 | | |
| | | 126 | | /// <summary> |
| | | 127 | | /// Filter workflow instances by timestamp. |
| | | 128 | | /// </summary> |
| | 252 | 129 | | public ICollection<TimestampFilter>? TimestampFilters { get; set; } |
| | | 130 | | |
| | | 131 | | /// <summary> |
| | | 132 | | /// Filter workflow instances by name. |
| | | 133 | | /// </summary> |
| | 242 | 134 | | public List<string>? Names { get; set; } |
| | | 135 | | |
| | | 136 | | /// <summary> |
| | | 137 | | /// Applies the filter to the specified query. |
| | | 138 | | /// </summary> |
| | | 139 | | [RequiresUnreferencedCode("The method uses reflection to create an expression tree.")] |
| | | 140 | | public IQueryable<WorkflowInstance> Apply(IQueryable<WorkflowInstance> query) |
| | | 141 | | { |
| | 242 | 142 | | var filter = this; |
| | | 143 | | |
| | 363 | 144 | | if (!string.IsNullOrWhiteSpace(filter.Id)) query = query.Where(x => x.Id == filter.Id); |
| | 252 | 145 | | if (filter.Ids != null) query = query.Where(x => filter.Ids.Contains(x.Id)); |
| | 263 | 146 | | if (!string.IsNullOrWhiteSpace(filter.DefinitionId)) query = query.Where(x => x.DefinitionId == filter.Definitio |
| | 242 | 147 | | if (!string.IsNullOrWhiteSpace(filter.DefinitionVersionId)) query = query.Where(x => x.DefinitionVersionId == fi |
| | 243 | 148 | | if (filter.DefinitionIds != null) query = query.Where(x => filter.DefinitionIds.Contains(x.DefinitionId)); |
| | 242 | 149 | | if (filter.DefinitionVersionIds != null) query = query.Where(x => filter.DefinitionVersionIds.Contains(x.Definit |
| | 242 | 150 | | if (filter.Version != null) query = query.Where(x => x.Version == filter.Version); |
| | 248 | 151 | | if (filter.ParentWorkflowInstanceIds != null) query = query.Where(x => x.ParentWorkflowInstanceId != null && fil |
| | 245 | 152 | | if (!string.IsNullOrWhiteSpace(filter.CorrelationId)) query = query.Where(x => x.CorrelationId == filter.Correla |
| | 242 | 153 | | if (filter.CorrelationIds != null) query = query.Where(x => filter.CorrelationIds.Contains(x.CorrelationId!)); |
| | 242 | 154 | | if (filter.Names != null) query = query.Where(x => filter.Names.Contains(x.Name!)); |
| | 328 | 155 | | if (filter.WorkflowStatus != null) query = query.Where(x => x.Status == filter.WorkflowStatus); |
| | 255 | 156 | | if (filter.WorkflowSubStatus != null) query = query.Where(x => x.SubStatus == filter.WorkflowSubStatus); |
| | 242 | 157 | | if (filter.WorkflowStatuses != null) query = query.Where(x => filter.WorkflowStatuses.Contains(x.Status)); |
| | 242 | 158 | | if (filter.WorkflowSubStatuses != null) query = query.Where(x => filter.WorkflowSubStatuses.Contains(x.SubStatus |
| | 314 | 159 | | if (filter.IsExecuting != null) query = query.Where(x => x.IsExecuting == filter.IsExecuting); |
| | 242 | 160 | | if (filter.HasIncidents != null) query = filter.HasIncidents == true ? query.Where(x => x.IncidentCount > 0) : q |
| | 242 | 161 | | if (filter.IsSystem != null) query = query.Where(x => x.IsSystem == filter.IsSystem); |
| | 242 | 162 | | if (filter.Name != null) query = query.Where(x => x.Name!.ToLower().Contains(filter.Name.ToLower())); |
| | 314 | 163 | | if (filter.BeforeLastUpdated != null) query = query.Where(x => x.UpdatedAt < filter.BeforeLastUpdated); |
| | | 164 | | |
| | 242 | 165 | | if (TimestampFilters != null) |
| | | 166 | | { |
| | 18 | 167 | | foreach (var timestampFilter in TimestampFilters) |
| | | 168 | | { |
| | 5 | 169 | | if (timestampFilter == null) |
| | 1 | 170 | | throw new ArgumentException("Timestamp filter must be specified.", nameof(TimestampFilters)); |
| | | 171 | | |
| | 4 | 172 | | var column = NormalizeTimestampFilterColumn(timestampFilter.Column); |
| | 3 | 173 | | var timestamp = timestampFilter.Timestamp; |
| | 3 | 174 | | var isZeroTime = timestamp.TimeOfDay == TimeSpan.Zero; |
| | 3 | 175 | | var startDay = new DateTimeOffset(timestamp.Date); |
| | 3 | 176 | | var endDay = startDay.AddDays(1); |
| | | 177 | | |
| | 3 | 178 | | query = timestampFilter.Operator switch |
| | 3 | 179 | | { |
| | 0 | 180 | | TimestampFilterOperator.Is when isZeroTime => query.Where($"{column} >= @0 && {column} < @1", startD |
| | 0 | 181 | | TimestampFilterOperator.Is => query.Where($"{column} == @0", timestamp), |
| | 0 | 182 | | TimestampFilterOperator.IsNot when isZeroTime => query.Where($"{column} < @0 || {column} >= @1", sta |
| | 0 | 183 | | TimestampFilterOperator.IsNot => query.Where($"{column} != @0", timestamp), |
| | 0 | 184 | | TimestampFilterOperator.GreaterThan when isZeroTime => query.Where($"{column} > @0", endDay), |
| | 0 | 185 | | TimestampFilterOperator.GreaterThan => query.Where($"{column} > @0", timestamp), |
| | 3 | 186 | | TimestampFilterOperator.GreaterThanOrEqual when isZeroTime => query.Where($"{column} >= @0", startDa |
| | 3 | 187 | | TimestampFilterOperator.GreaterThanOrEqual => query.Where($"{column} >= @0", timestamp), |
| | 0 | 188 | | TimestampFilterOperator.LessThan when isZeroTime => query.Where($"{column} < @0", startDay), |
| | 0 | 189 | | TimestampFilterOperator.LessThan => query.Where($"{column} < @0", timestamp), |
| | 0 | 190 | | TimestampFilterOperator.LessThanOrEqual when isZeroTime => query.Where($"{column} <= @0", endDay), |
| | 0 | 191 | | TimestampFilterOperator.LessThanOrEqual => query.Where($"{column} <= @0", timestamp), |
| | 0 | 192 | | _ => query |
| | 3 | 193 | | }; |
| | | 194 | | } |
| | | 195 | | } |
| | | 196 | | |
| | 240 | 197 | | var searchTerm = filter.SearchTerm; |
| | 240 | 198 | | if (!string.IsNullOrWhiteSpace(searchTerm)) |
| | | 199 | | { |
| | 0 | 200 | | query = |
| | 0 | 201 | | from instance in query |
| | 0 | 202 | | where instance.Name!.ToLower().Contains(searchTerm.ToLower()) |
| | 0 | 203 | | || instance.DefinitionVersionId.Contains(searchTerm) |
| | 0 | 204 | | || instance.DefinitionId.Contains(searchTerm) |
| | 0 | 205 | | || instance.Id.Contains(searchTerm) |
| | 0 | 206 | | || instance.CorrelationId!.Contains(searchTerm) |
| | 0 | 207 | | select instance; |
| | | 208 | | } |
| | | 209 | | |
| | 240 | 210 | | return query; |
| | | 211 | | } |
| | | 212 | | |
| | | 213 | | /// <summary> |
| | | 214 | | /// Validates timestamp filters. |
| | | 215 | | /// </summary> |
| | | 216 | | public static IEnumerable<string> ValidateTimestampFilters(IEnumerable<TimestampFilter>? timestampFilters) |
| | | 217 | | { |
| | 2 | 218 | | if (timestampFilters == null) |
| | 0 | 219 | | yield break; |
| | | 220 | | |
| | 10 | 221 | | foreach (var (timestampFilter, index) in timestampFilters.Select((value, index) => (value, index))) |
| | | 222 | | { |
| | 2 | 223 | | if (timestampFilter == null) |
| | | 224 | | { |
| | 1 | 225 | | yield return $"Timestamp filter at index {index} must be specified."; |
| | 1 | 226 | | continue; |
| | | 227 | | } |
| | | 228 | | |
| | 1 | 229 | | if (!TryNormalizeTimestampFilterColumn(timestampFilter.Column, out _, out var error)) |
| | 1 | 230 | | yield return $"Timestamp filter at index {index}: {error}"; |
| | | 231 | | } |
| | 2 | 232 | | } |
| | | 233 | | |
| | | 234 | | /// <summary> |
| | | 235 | | /// Resolves a timestamp filter column to its canonical workflow instance property name. |
| | | 236 | | /// </summary> |
| | | 237 | | public static bool TryNormalizeTimestampFilterColumn(string? column, [NotNullWhen(true)] out string? normalizedColum |
| | | 238 | | { |
| | 5 | 239 | | normalizedColumn = null; |
| | 5 | 240 | | error = null; |
| | | 241 | | |
| | 5 | 242 | | if (string.IsNullOrWhiteSpace(column)) |
| | | 243 | | { |
| | 1 | 244 | | error = "Timestamp filter column must be specified."; |
| | 1 | 245 | | return false; |
| | | 246 | | } |
| | | 247 | | |
| | 4 | 248 | | var trimmedColumn = column.Trim(); |
| | 13 | 249 | | normalizedColumn = TimestampFilterColumns.FirstOrDefault(x => string.Equals(x, trimmedColumn, StringComparison.O |
| | | 250 | | |
| | 4 | 251 | | if (normalizedColumn != null) |
| | 3 | 252 | | return true; |
| | | 253 | | |
| | 1 | 254 | | error = $"Invalid timestamp filter column. Allowed columns are: {string.Join(", ", TimestampFilterColumns)}."; |
| | 1 | 255 | | return false; |
| | | 256 | | } |
| | | 257 | | |
| | | 258 | | private static string NormalizeTimestampFilterColumn(string? column) |
| | | 259 | | { |
| | 4 | 260 | | if (TryNormalizeTimestampFilterColumn(column, out var normalizedColumn, out var error)) |
| | 3 | 261 | | return normalizedColumn; |
| | | 262 | | |
| | 1 | 263 | | throw new ArgumentException(error, $"{nameof(TimestampFilters)}.Column"); |
| | | 264 | | } |
| | | 265 | | } |