protected bool AnalyzeClause(Clause clause, object?before = null, object?after = null) { if (clause == null || (before == null && after == null)) { return(false); } try { // Support bare objects if (clause.Field is string) { after = GetValueByPropertyString(after, clause.Field); before = GetValueByPropertyString(before, clause.Field); } var typeHolder = before is null ? after : before; (var beforeList, var beforeDict) = ObjectToValues(before); (var afterList, var afterDict) = ObjectToValues(after); var valsToCheck = beforeList.Union(afterList); var dictToCheck = beforeDict.Union(afterDict); switch (clause.Operation) { case OPERATION.EQ: if (clause.Data is List <string> EqualsData) { if (EqualsData.Intersect(valsToCheck).Any()) { return(true); } } return(false); case OPERATION.NEQ: if (clause.Data is List <string> NotEqualsData) { if (!NotEqualsData.Intersect(valsToCheck).Any()) { return(true); } } return(false); // If *every* entry of the clause data is matched case OPERATION.CONTAINS: if (dictToCheck.Any()) { if (clause.DictData is List <KeyValuePair <string, string> > ContainsData) { if (ContainsData.All(y => dictToCheck.Any((x) => x.Key == y.Key && x.Value == y.Value))) { return(true); } } } else if (valsToCheck.Any()) { if (clause.Data is List <string> ContainsDataList) { // If we are dealing with an array on the object side if (typeHolder is List <string> ) { if (ContainsDataList.All(x => valsToCheck.Contains(x))) { return(true); } } // If we are dealing with a single string we do a .Contains instead else if (typeHolder is string) { if (clause.Data.All(x => valsToCheck.First()?.Contains(x) ?? false)) { return(true); } } } } return(false); // If *any* entry of the clause data is matched case OPERATION.CONTAINS_ANY: if (dictToCheck.Any()) { if (clause.DictData is List <KeyValuePair <string, string> > ContainsData) { foreach (KeyValuePair <string, string> value in ContainsData) { if (dictToCheck.Any(x => x.Key == value.Key && x.Value == value.Value)) { return(true); } } } } else if (valsToCheck.Any()) { if (clause.Data is List <string> ContainsDataList) { if (typeHolder is List <string> ) { if (ContainsDataList.Any(x => valsToCheck.Contains(x))) { return(true); } } // If we are dealing with a single string we do a .Contains instead else if (typeHolder is string) { if (clause.Data.Any(x => valsToCheck.First()?.Contains(x) ?? false)) { return(true); } } } } return(false); // If any of the data values are greater than the first provided clause value We // ignore all other clause values case OPERATION.GT: foreach (var val in valsToCheck) { if (int.TryParse(val, out int valToCheck)) { if (int.TryParse(clause.Data?[0], out int dataValue)) { if (valToCheck > dataValue) { return(true); } } } } return(false); // If any of the data values are less than the first provided clause value We // ignore all other clause values case OPERATION.LT: foreach (var val in valsToCheck) { if (int.TryParse(val, out int valToCheck)) { if (int.TryParse(clause.Data?[0], out int dataValue)) { if (valToCheck < dataValue) { return(true); } } } } return(false); // If any of the regexes match any of the values case OPERATION.REGEX: if (clause.Data is List <string> RegexList) { if (RegexList.Count > 0) { var built = string.Join('|', RegexList); if (!RegexCache.ContainsKey(built)) { try { RegexCache.TryAdd(built, new Regex(built, RegexOptions.Compiled)); } catch (ArgumentException) { Log.Warning("InvalidArgumentException when analyzing clause {0}. Regex {1} is invalid and will be skipped.", clause.Label, built); RegexCache.TryAdd(built, new Regex("", RegexOptions.Compiled)); } } if (valsToCheck.Any(x => RegexCache[built].IsMatch(x))) { return(true); } } } return(false); // Ignores provided data. Checks if the named property has changed. case OPERATION.WAS_MODIFIED: CompareLogic compareLogic = new CompareLogic(); ComparisonResult comparisonResult = compareLogic.Compare(before, after); return(!comparisonResult.AreEqual); // Ends with any of the provided data case OPERATION.ENDS_WITH: if (clause.Data is List <string> EndsWithData) { if (valsToCheck.Any(x => EndsWithData.Any(y => x is string && x.EndsWith(y, StringComparison.CurrentCulture)))) { return(true); } } return(false); // Starts with any of the provided data case OPERATION.STARTS_WITH: if (clause.Data is List <string> StartsWithData) { if (valsToCheck.Any(x => StartsWithData.Any(y => x is string && x.StartsWith(y, StringComparison.CurrentCulture)))) { return(true); } } return(false); case OPERATION.IS_NULL: if (valsToCheck.Count(x => x is null) == valsToCheck.Count()) { return(true); } return(false); case OPERATION.IS_TRUE: foreach (var valToCheck in valsToCheck) { if (bool.TryParse(valToCheck, out bool result)) { if (result) { return(true); } } } return(false); case OPERATION.IS_BEFORE: var valDateTimes = new List <DateTime>(); foreach (var valToCheck in valsToCheck) { if (DateTime.TryParse(valToCheck, out DateTime result)) { valDateTimes.Add(result); } } foreach (var data in clause.Data ?? new List <string>()) { if (DateTime.TryParse(data, out DateTime result)) { if (valDateTimes.Any(x => x.CompareTo(result) < 0)) { return(true); } } } return(false); case OPERATION.IS_AFTER: valDateTimes = new List <DateTime>(); foreach (var valToCheck in valsToCheck) { if (DateTime.TryParse(valToCheck, out DateTime result)) { valDateTimes.Add(result); } } foreach (var data in clause.Data ?? new List <string>()) { if (DateTime.TryParse(data, out DateTime result)) { if (valDateTimes.Any(x => x.CompareTo(result) > 0)) { return(true); } } } return(false); case OPERATION.IS_EXPIRED: foreach (var valToCheck in valsToCheck) { if (DateTime.TryParse(valToCheck, out DateTime result)) { if (result.CompareTo(DateTime.Now) < 0) { return(true); } } } return(false); case OPERATION.CONTAINS_KEY: return(dictToCheck.Any(x => clause.Data.Any(y => x.Key == y))); case OPERATION.CUSTOM: if (CustomOperationDelegate is null) { Log.Debug("Custom operation hit but {0} isn't set.", nameof(CustomOperationDelegate)); return(false); } else { return(CustomOperationDelegate.Invoke(clause, valsToCheck, dictToCheck)); } default: Log.Debug("Unimplemented operation {0}", clause.Operation); return(false); } } catch (Exception e) { Log.Debug(e, $"Hit while parsing {JsonConvert.SerializeObject(clause)} onto ({JsonConvert.SerializeObject(before)},{JsonConvert.SerializeObject(after)})"); } return(false); }
protected ANALYSIS_RESULT_TYPE Apply(Rule rule, CompareResult compareResult) { if (compareResult != null && rule != null) { var properties = _Properties[compareResult.ResultType]; foreach (Clause clause in rule.Clauses) { PropertyInfo property = properties.FirstOrDefault(iProp => iProp.Name.Equals(clause.Field)); if (property == null) { //Custom field logic will go here return(DEFAULT_RESULT_TYPE_MAP[compareResult.ResultType]); } try { var valsToCheck = new List <string>(); List <KeyValuePair <string, string> > dictToCheck = new List <KeyValuePair <string, string> >(); if (property != null) { if (compareResult.ChangeType == CHANGE_TYPE.CREATED || compareResult.ChangeType == CHANGE_TYPE.MODIFIED) { try { if (GetValueByPropertyName(compareResult.Compare, property.Name) is List <string> ) { foreach (var value in (List <string>)(GetValueByPropertyName(compareResult.Compare, property.Name) ?? new List <string>())) { valsToCheck.Add(value); } } else if (GetValueByPropertyName(compareResult.Compare, property.Name) is Dictionary <string, string> ) { dictToCheck = ((Dictionary <string, string>)(GetValueByPropertyName(compareResult.Compare, property.Name) ?? new Dictionary <string, string>())).ToList(); } else if (GetValueByPropertyName(compareResult.Compare, property.Name) is List <KeyValuePair <string, string> > ) { dictToCheck = (List <KeyValuePair <string, string> >)(GetValueByPropertyName(compareResult.Compare, property.Name) ?? new List <KeyValuePair <string, string> >()); } else { var val = GetValueByPropertyName(compareResult.Compare, property.Name)?.ToString(); if (!string.IsNullOrEmpty(val)) { valsToCheck.Add(val); } } } catch (Exception e) { Log.Debug(e, "Error fetching Property {0} of Type {1}", property.Name, compareResult.ResultType); Log.Debug(Utf8Json.JsonSerializer.ToJsonString(compareResult)); Dictionary <string, string> ExceptionEvent = new Dictionary <string, string>(); ExceptionEvent.Add("Exception Type", e.GetType().ToString()); AsaTelemetry.TrackEvent("ApplyCreatedModifiedException", ExceptionEvent); } } if (compareResult.ChangeType == CHANGE_TYPE.DELETED || compareResult.ChangeType == CHANGE_TYPE.MODIFIED) { try { if (GetValueByPropertyName(compareResult.Base, property.Name) is List <string> ) { foreach (var value in (List <string>)(GetValueByPropertyName(compareResult.Base, property.Name) ?? new List <string>())) { valsToCheck.Add(value); } } else if (GetValueByPropertyName(compareResult.Base, property.Name) is Dictionary <string, string> ) { dictToCheck = ((Dictionary <string, string>)(GetValueByPropertyName(compareResult.Base, property.Name) ?? new Dictionary <string, string>())).ToList(); } else if (GetValueByPropertyName(compareResult.Base, property.Name) is List <KeyValuePair <string, string> > ) { dictToCheck = (List <KeyValuePair <string, string> >)(GetValueByPropertyName(compareResult.Base, property.Name) ?? new List <KeyValuePair <string, string> >()); } else { var val = GetValueByPropertyName(compareResult.Base, property.Name)?.ToString(); if (!string.IsNullOrEmpty(val)) { valsToCheck.Add(val); } } } catch (Exception e) { Dictionary <string, string> ExceptionEvent = new Dictionary <string, string>(); ExceptionEvent.Add("Exception Type", e.GetType().ToString()); AsaTelemetry.TrackEvent("ApplyDeletedModifiedException", ExceptionEvent); } } } switch (clause.Operation) { case OPERATION.EQ: if (clause.Data is List <string> EqualsData) { if (EqualsData.Intersect(valsToCheck).Any()) { break; } } return(DEFAULT_RESULT_TYPE_MAP[compareResult.ResultType]); case OPERATION.NEQ: if (clause.Data is List <string> NotEqualsData) { if (!NotEqualsData.Intersect(valsToCheck).Any()) { break; } } return(DEFAULT_RESULT_TYPE_MAP[compareResult.ResultType]); // If *every* entry of the clause data is matched case OPERATION.CONTAINS: if (dictToCheck.Count > 0) { if (clause.DictData is List <KeyValuePair <string, string> > ContainsData) { if (ContainsData.Where(y => dictToCheck.Where((x) => x.Key == y.Key && x.Value == y.Value).Any()).Count() == ContainsData.Count) { break; } } } else if (valsToCheck.Count > 0) { if (clause.Data is List <string> ContainsDataList) { if (ContainsDataList.Intersect(valsToCheck).Count() == ContainsDataList.Count) { break; } } } return(DEFAULT_RESULT_TYPE_MAP[compareResult.ResultType]); // If *any* entry of the clause data is matched case OPERATION.CONTAINS_ANY: if (dictToCheck.Count > 0) { if (clause.DictData is List <KeyValuePair <string, string> > ContainsData) { foreach (KeyValuePair <string, string> value in ContainsData) { if (dictToCheck.Where((x) => x.Key == value.Key && x.Value == value.Value).Any()) { break; } } } } else if (valsToCheck.Count > 0) { if (clause.Data is List <string> ContainsDataList) { if (clause.Data.Intersect(valsToCheck).Any()) { break; } } } return(DEFAULT_RESULT_TYPE_MAP[compareResult.ResultType]); // If any of the clauses are not contained case OPERATION.DOES_NOT_CONTAIN: if (dictToCheck.Count > 0) { if (clause.DictData is List <KeyValuePair <string, string> > ContainsData) { if (ContainsData.Where(y => dictToCheck.Where((x) => x.Key == y.Key && x.Value == y.Value).Any()).Any()) { break; } return(DEFAULT_RESULT_TYPE_MAP[compareResult.ResultType]); } } else if (valsToCheck.Count > 0) { if (clause.Data is List <string> ContainsDataList) { if (ContainsDataList.Intersect(valsToCheck).Any()) { break; } return(DEFAULT_RESULT_TYPE_MAP[compareResult.ResultType]); } } break; // If any of the data values are greater than the first provided data value case OPERATION.GT: if (valsToCheck.Where(val => (int.Parse(val, CultureInfo.InvariantCulture) > int.Parse(clause.Data?[0] ?? $"{int.MinValue}", CultureInfo.InvariantCulture))).Any()) { break; } return(DEFAULT_RESULT_TYPE_MAP[compareResult.ResultType]); // If any of the data values are less than the first provided data value case OPERATION.LT: if (valsToCheck.Where(val => (int.Parse(val, CultureInfo.InvariantCulture) < int.Parse(clause.Data?[0] ?? $"{int.MaxValue}", CultureInfo.InvariantCulture))).Any()) { break; } return(DEFAULT_RESULT_TYPE_MAP[compareResult.ResultType]); // If any of the regexes match any of the values case OPERATION.REGEX: if (clause.Data is List <string> RegexList) { var regexList = RegexList.Select(x => new Regex(x)); if (valsToCheck.Where(x => regexList.Where(y => y.IsMatch(x)).Any()).Any()) { break; } } return(DEFAULT_RESULT_TYPE_MAP[compareResult.ResultType]); // Ignores provided data. Checks if the named property has changed. case OPERATION.WAS_MODIFIED: if ((valsToCheck.Count == 2) && (valsToCheck[0] == valsToCheck[1])) { break; } return(DEFAULT_RESULT_TYPE_MAP[compareResult.ResultType]); // Ends with any of the provided data case OPERATION.ENDS_WITH: if (clause.Data is List <string> EndsWithData) { if (valsToCheck.Where(x => EndsWithData.Where(y => x.EndsWith(y, StringComparison.CurrentCulture)).Any()).Any()) { break; } } return(DEFAULT_RESULT_TYPE_MAP[compareResult.ResultType]); // Starts with any of the provided data case OPERATION.STARTS_WITH: if (clause.Data is List <string> StartsWithData) { if (valsToCheck.Where(x => StartsWithData.Where(y => x.StartsWith(y, StringComparison.CurrentCulture)).Any()).Any()) { break; } } return(DEFAULT_RESULT_TYPE_MAP[compareResult.ResultType]); default: Log.Debug("Unimplemented operation {0}", clause.Operation); return(DEFAULT_RESULT_TYPE_MAP[compareResult.ResultType]); } } catch (Exception e) { Log.Debug(e, $"Hit while parsing {JsonSerializer.Serialize(rule)} onto {JsonSerializer.Serialize(compareResult)}"); Dictionary <string, string> ExceptionEvent = new Dictionary <string, string>(); ExceptionEvent.Add("Exception Type", e.GetType().ToString()); AsaTelemetry.TrackEvent("ApplyOverallException", ExceptionEvent); return(DEFAULT_RESULT_TYPE_MAP[compareResult.ResultType]); } } compareResult.Rules.Add(rule); return(rule.Flag); } else { throw new NullReferenceException(); } }