using System; using Xunit; using Newtonsoft.Json.Linq; using System.Linq; using FluentAssertions; using System.Collections.Generic; namespace raven_integration { public class SearchOps { //NOTE: this test failed one time for no apparent reason but then was ok afterwards [Fact] public async void PhraseOnlySearchShouldReturnCorrectResultsInOrder() { const string TEST_SEARCH_PHRASE = "simple dogs"; //CREATE A WIDGET dynamic D = new JObject(); D.name = Util.Uniquify("search NOTES Test WIDGET"); D.customFields = Util.WidgetRequiredCustomFieldsJsonString(); D.dollarAmount = 1.11m; D.active = true; D.usertype = 1; D.notes = "This record will match in notes: The quick brown and simple fox jumped over the six lazy dogs!"; ApiResponse a = await Util.PostAsync("widget", await Util.GetTokenAsync("superuser", "l3tm3in"), D.ToString()); Util.ValidateDataReturnResponseOk(a); long MatchWidgetInNotesId = a.ObjectResponse["data"]["id"].Value(); //CREATE FIRST TEST USER WITH PHRASE IN NAME D = new JObject(); D.name = Util.Uniquify("search NAME DOGS simple Test User"); D.customFields = Util.WidgetRequiredCustomFieldsJsonString(); D.notes = "This user has the match in it's name"; D.active = true; D.login = Util.Uniquify("LOGIN"); D.password = Util.Uniquify("PASSWORD"); D.usertype = 1; D.userType = 3;//non scheduleable a = await Util.PostAsync("User", await Util.GetTokenAsync("superuser", "l3tm3in"), D.ToString()); Util.ValidateDataReturnResponseOk(a); long MatchUserInNameId = a.ObjectResponse["data"]["id"].Value(); //CREATE A SECOND TEST USER WITH PHRASE IN NOTES D = new JObject(); D.name = Util.Uniquify("search NOTES Test User"); D.customFields = Util.WidgetRequiredCustomFieldsJsonString(); D.notes = "This user has the match simple dogs in its notes"; D.active = true; D.login = Util.Uniquify("LOGIN"); D.password = Util.Uniquify("PASSWORD"); D.usertype = 1; D.userType = 3;//non scheduleable a = await Util.PostAsync("User", await Util.GetTokenAsync("superuser", "l3tm3in"), D.ToString()); Util.ValidateDataReturnResponseOk(a); long MatchUserInNotesId = a.ObjectResponse["data"]["id"].Value(); //CREATE A SECOND WIDGET D = new JObject(); D.name = Util.Uniquify("search NAME simple as in dogs Test WIDGET"); D.customFields = Util.WidgetRequiredCustomFieldsJsonString(); D.dollarAmount = 1.11m; D.active = true; D.usertype = 1; D.notes = "This Widget matches in name"; a = await Util.PostAsync("widget", await Util.GetTokenAsync("superuser", "l3tm3in"), D.ToString()); Util.ValidateDataReturnResponseOk(a); long MatchWidgetInNameId = a.ObjectResponse["data"]["id"].Value(); //CREATE A THIRD WIDGET D = new JObject(); D.name = Util.Uniquify("search NO-MATCH THIRD Test WIDGET"); D.customFields = Util.WidgetRequiredCustomFieldsJsonString(); D.dollarAmount = 1.11m; D.active = true; D.usertype = 1; D.notes = "This Widget should not be returned in the search as it only contains a single keyword in the name not both"; a = await Util.PostAsync("widget", await Util.GetTokenAsync("superuser", "l3tm3in"), D.ToString()); Util.ValidateDataReturnResponseOk(a); long MatchNothingWidgetId = a.ObjectResponse["data"]["id"].Value(); //Now see if can find those objects with a phrase search dynamic SearchParameters = new JObject(); SearchParameters.phrase = TEST_SEARCH_PHRASE; SearchParameters.typeOnly = 0;//no type SearchParameters.maxResults = 0; a = await Util.PostAsync("search", await Util.GetTokenAsync("superuser", "l3tm3in"), SearchParameters.ToString()); Util.ValidateDataReturnResponseOk(a); //Now validate the return list ((JArray)a.ObjectResponse["data"]["searchResults"]).Count.Should().BeGreaterOrEqualTo(3); //Turn the list into an array of id's var v = ((JArray)a.ObjectResponse["data"]["searchResults"]); List MatchingIdList = new List(); foreach (JObject j in v) { MatchingIdList.Add(j["id"].Value()); } //Ensure the expected items are returned MatchingIdList.Should().Contain(MatchWidgetInNotesId, "ShouldContainMatchWidgetInNotesId"); MatchingIdList.Should().Contain(MatchWidgetInNameId, "ShouldContainMatchWidgetInNameId"); MatchingIdList.Should().Contain(MatchUserInNotesId, "ShouldContainMatchUserInNotesId"); MatchingIdList.Should().Contain(MatchUserInNameId, "ShouldContainMatchUserInNameId"); MatchingIdList.Should().NotContain(MatchNothingWidgetId, "ShouldNotContainMatchNothingWidgetId"); //Assert the order (roughly, this is kind of a waste of time, either the code is sorting or not, it's not going to change) //first item must be a widget a.ObjectResponse["data"]["searchResults"][0]["type"].Value().Should().Be(2); //final item must be a user a.ObjectResponse["data"]["searchResults"][MatchingIdList.Count - 1]["type"].Value().Should().Be(3); //FULL BODY SEARCH RIGHTS //First up test a full record search returns no results due to insufficient rights //even though the record exists //Just re-run the above search exactly but with a no rights to full User or Widget role instead //Only BizAdmin* roles can read a full user record //This search should return zero items a = await Util.PostAsync("search", await Util.GetTokenAsync("SubContractorLimited"), SearchParameters.ToString()); Util.ValidateDataReturnResponseOk(a); ((JArray)a.ObjectResponse["data"]["searchResults"]).Count.Should().Be(0, "User with no rights should not see any results in body search"); }//eot [Fact] public async void WildCardStartsWithSearchShouldWork() { const string TEST_SEARCH_PHRASE = "hap* goose"; //CREATE A WIDGET dynamic D = new JObject(); D.name = Util.Uniquify("Wildcard startswith search test WIDGET"); D.customFields = Util.WidgetRequiredCustomFieldsJsonString(); D.dollarAmount = 1.11m; D.active = true; D.usertype = 1; D.notes = "This record will match in notes: The quick brown and hapless goose"; ApiResponse a = await Util.PostAsync("widget", await Util.GetTokenAsync("superuser", "l3tm3in"), D.ToString()); Util.ValidateDataReturnResponseOk(a); long MatchWidgetInNotesId = a.ObjectResponse["data"]["id"].Value(); //CREATE FIRST TEST USER WITH PHRASE IN NAME D = new JObject(); D.name = Util.Uniquify("Wildcard startswith search NAME happy goose Test User"); D.notes = "This user has the match in it's name"; D.active = true; D.login = Util.Uniquify("LOGIN"); D.password = Util.Uniquify("PASSWORD"); D.usertype = 1; D.userType = 3;//non scheduleable D.customFields = Util.UserRequiredCustomFieldsJsonString(); a = await Util.PostAsync("User", await Util.GetTokenAsync("superuser", "l3tm3in"), D.ToString()); Util.ValidateDataReturnResponseOk(a); long MatchUserInNameId = a.ObjectResponse["data"]["id"].Value(); //Now see if can find those objects with a phrase search dynamic SearchParameters = new JObject(); SearchParameters.phrase = TEST_SEARCH_PHRASE; SearchParameters.typeOnly = 0;//no type SearchParameters.maxResults = 0; a = await Util.PostAsync("search", await Util.GetTokenAsync("superuser", "l3tm3in"), SearchParameters.ToString()); Util.ValidateDataReturnResponseOk(a); //Now validate the return list ((JArray)a.ObjectResponse["data"]["searchResults"]).Count.Should().BeGreaterOrEqualTo(2); //Turn the list into an array of id's var v = ((JArray)a.ObjectResponse["data"]["searchResults"]); List MatchingIdList = new List(); foreach (JObject j in v) { MatchingIdList.Add(j["id"].Value()); } //Ensure the expected items are returned MatchingIdList.Should().Contain(MatchWidgetInNotesId, "ShouldContainMatchWidgetInNotesId"); MatchingIdList.Should().Contain(MatchUserInNameId, "ShouldContainMatchUserInNameId"); }//eot [Fact] public async void WildCardEndsWithSearchShouldWork() { const string TEST_SEARCH_PHRASE = "goose *act"; //CREATE A WIDGET dynamic D = new JObject(); D.name = Util.Uniquify("Wildcard endswith search test WIDGET"); D.customFields = Util.WidgetRequiredCustomFieldsJsonString(); D.dollarAmount = 1.11m; D.active = true; D.usertype = 1; D.notes = "This record will match in notes: react to the goose"; ApiResponse a = await Util.PostAsync("widget", await Util.GetTokenAsync("superuser", "l3tm3in"), D.ToString()); Util.ValidateDataReturnResponseOk(a); long MatchWidgetInNotesId = a.ObjectResponse["data"]["id"].Value(); //CREATE FIRST TEST USER WITH PHRASE IN NAME D = new JObject(); D.name = Util.Uniquify("Wildcard endswith search NAME goose exact Test User"); D.notes = "This user has the match in it's name"; D.active = true; D.login = Util.Uniquify("LOGIN"); D.password = Util.Uniquify("PASSWORD"); D.usertype = 1; D.userType = 3;//non scheduleable D.customFields = Util.UserRequiredCustomFieldsJsonString(); a = await Util.PostAsync("User", await Util.GetTokenAsync("superuser", "l3tm3in"), D.ToString()); Util.ValidateDataReturnResponseOk(a); long MatchUserInNameId = a.ObjectResponse["data"]["id"].Value(); //Now see if can find those objects with a phrase search dynamic SearchParameters = new JObject(); SearchParameters.phrase = TEST_SEARCH_PHRASE; SearchParameters.typeOnly = 0;//no type SearchParameters.maxResults = 0; a = await Util.PostAsync("search", await Util.GetTokenAsync("superuser", "l3tm3in"), SearchParameters.ToString()); Util.ValidateDataReturnResponseOk(a); //Now validate the return list ((JArray)a.ObjectResponse["data"]["searchResults"]).Count.Should().BeGreaterOrEqualTo(2); //Turn the list into an array of id's var v = ((JArray)a.ObjectResponse["data"]["searchResults"]); List MatchingIdList = new List(); foreach (JObject j in v) { MatchingIdList.Add(j["id"].Value()); } //Ensure the expected items are returned MatchingIdList.Should().Contain(MatchWidgetInNotesId, "ShouldContainMatchWidgetInNotesId"); MatchingIdList.Should().Contain(MatchUserInNameId, "ShouldContainMatchUserInNameId"); }//eot [Fact] public async void SearchAndExtractShouldWork() { const string TEST_SEARCH_PHRASE = "cast goose"; //CREATE A WIDGET dynamic D = new JObject(); D.name = Util.Uniquify("Wildcard contains search test WIDGET"); D.customFields = Util.WidgetRequiredCustomFieldsJsonString(); D.dollarAmount = 1.11m; D.active = true; D.usertype = 1; D.notes = "This record will match in notes: broadcasting to the goose"; ApiResponse a = await Util.PostAsync("widget", await Util.GetTokenAsync("superuser", "l3tm3in"), D.ToString()); Util.ValidateDataReturnResponseOk(a); long MatchWidgetInNotesId = a.ObjectResponse["data"]["id"].Value(); //CREATE FIRST TEST USER WITH PHRASE IN NAME D = new JObject(); D.name = Util.Uniquify("Wildcard contains search NAME goose elcastro Test User"); D.notes = "This user has the match in it's name"; D.active = true; D.login = Util.Uniquify("LOGIN"); D.password = Util.Uniquify("PASSWORD"); D.usertype = 1; D.userType = 3;//non scheduleable D.customFields = Util.UserRequiredCustomFieldsJsonString(); a = await Util.PostAsync("User", await Util.GetTokenAsync("superuser", "l3tm3in"), D.ToString()); Util.ValidateDataReturnResponseOk(a); long MatchUserInNameId = a.ObjectResponse["data"]["id"].Value(); //Now see if can find those objects with a phrase search dynamic SearchParameters = new JObject(); SearchParameters.phrase = TEST_SEARCH_PHRASE; SearchParameters.typeOnly = 0;//no type SearchParameters.maxResults = 0; a = await Util.PostAsync("search", await Util.GetTokenAsync("superuser", "l3tm3in"), SearchParameters.ToString()); Util.ValidateDataReturnResponseOk(a); //Now validate the return list ((JArray)a.ObjectResponse["data"]["searchResults"]).Count.Should().BeGreaterOrEqualTo(2); //Turn the list into an array of id's var v = ((JArray)a.ObjectResponse["data"]["searchResults"]); List MatchingIdList = new List(); foreach (JObject j in v) { MatchingIdList.Add(j["id"].Value()); } //Ensure the expected items are returned MatchingIdList.Should().Contain(MatchWidgetInNotesId, "ShouldContainMatchWidgetInNotesId"); MatchingIdList.Should().Contain(MatchUserInNameId, "ShouldContainMatchUserInNameId"); //Ensure excerpt works and contains the search phrase // /Search/Info/2/1?phrase=we&max=200 a = await Util.GetAsync($"search/Info/2/{MatchWidgetInNotesId}?phrase={TEST_SEARCH_PHRASE}&max=200", await Util.GetTokenAsync("superuser", "l3tm3in")); Util.ValidateDataReturnResponseOk(a); a.ObjectResponse["data"].Value().Should().Contain(TEST_SEARCH_PHRASE.Split(" ")[0]); a.ObjectResponse["data"].Value().Should().Contain(TEST_SEARCH_PHRASE.Split(" ")[1]); }//eot [Fact] public async void TagSearchShouldWork() { const string TEST_SEARCH_PHRASE = "element* aardvark"; // //CREATE A TAG // dynamic D = new JObject(); // D.name = Util.Uniquify("TAGSEARCH"); // ApiResponse a = await Util.PostAsync("Tag", await Util.GetTokenAsync("superuser", "l3tm3in"), D.ToString()); // Util.ValidateDataReturnResponseOk(a); // long TagId = a.ObjectResponse["data"]["id"].Value(); var SearchTagPhrase = Util.Uniquify("TAGSEARCH"); //Tags dynamic TagsArray = new JArray(); TagsArray.Add(SearchTagPhrase); //w.tags = InclusiveTagsArray; //CREATE A WIDGET dynamic D = new JObject(); D.name = Util.Uniquify("TAG search test WIDGET TAG AND PHRASE"); D.customFields = Util.WidgetRequiredCustomFieldsJsonString(); D.dollarAmount = 1.11m; D.active = true; D.usertype = 1; D.notes = "This record will match in notes and tag: elementary my dear aardvark"; D.tags = TagsArray; ApiResponse a = await Util.PostAsync("widget", await Util.GetTokenAsync("superuser", "l3tm3in"), D.ToString()); Util.ValidateDataReturnResponseOk(a); long MatchWidgetInNotesId = a.ObjectResponse["data"]["id"].Value(); //CREATE A WIDGET WITH TAG BUT NOT SEARCH PHRASE D = new JObject(); D.name = Util.Uniquify("TAG search test WIDGET TAG ONLY"); D.customFields = Util.WidgetRequiredCustomFieldsJsonString(); D.dollarAmount = 1.11m; D.active = true; D.usertype = 1; D.notes = "This record will match in tag but no search phrase"; D.tags = TagsArray; a = await Util.PostAsync("widget", await Util.GetTokenAsync("superuser", "l3tm3in"), D.ToString()); Util.ValidateDataReturnResponseOk(a); long NoPhraseMatchWidgetInTagId = a.ObjectResponse["data"]["id"].Value(); //CREATE FIRST TEST USER WITH PHRASE IN NAME BUT NO TAG D = new JObject(); D.name = Util.Uniquify("Wildcard contains search NAME elementary aardvark Test User"); D.notes = "This user has the match in it's name but no tag match"; D.active = true; D.login = Util.Uniquify("LOGIN"); D.password = Util.Uniquify("PASSWORD"); D.usertype = 1; D.userType = 3;//non scheduleable D.customFields = Util.UserRequiredCustomFieldsJsonString(); a = await Util.PostAsync("User", await Util.GetTokenAsync("superuser", "l3tm3in"), D.ToString()); Util.ValidateDataReturnResponseOk(a); long MatchUserInNameId = a.ObjectResponse["data"]["id"].Value(); //Now see if can find those objects with a phrase search dynamic SearchParameters = new JObject(); SearchParameters.phrase = TEST_SEARCH_PHRASE + " " + SearchTagPhrase; SearchParameters.typeOnly = 0;//no type a = await Util.PostAsync("search", await Util.GetTokenAsync("superuser", "l3tm3in"), SearchParameters.ToString()); Util.ValidateDataReturnResponseOk(a); //Now validate the return list ((JArray)a.ObjectResponse["data"]["searchResults"]).Count.Should().BeGreaterOrEqualTo(1); //Turn the list into an array of id's var v = ((JArray)a.ObjectResponse["data"]["searchResults"]); List MatchingIdList = new List(); foreach (JObject j in v) { MatchingIdList.Add(j["id"].Value()); } //Ensure the expected items are returned MatchingIdList.Should().Contain(MatchWidgetInNotesId, "ShouldContainMatchWidgetInNotesId"); MatchingIdList.Should().NotContain(MatchUserInNameId, "ShouldNotContainMatchUserInNameId"); MatchingIdList.Should().NotContain(NoPhraseMatchWidgetInTagId, "ShouldNotContainNoPhraseMatchWidgetInTagId"); }//eot [Fact] public async void ConstrainedBigDataSearchShouldHonourMaxResultsAndBeRelativelyFast() { //THIS test is a bit different in that it relies partly on the big dataset for testing //so it has different paths depending upon if it's testing against the big data or not const string TEST_SEARCH_PHRASE = "et*"; //Now see if can find those objects with a phrase search dynamic SearchParameters = new JObject(); SearchParameters.phrase = TEST_SEARCH_PHRASE; SearchParameters.typeOnly = 0;//no type SearchParameters.maxResults = 1000;//default is 500 var watch = new System.Diagnostics.Stopwatch(); watch.Start(); ApiResponse a = await Util.PostAsync("search", await Util.GetTokenAsync("superuser", "l3tm3in"), SearchParameters.ToString()); watch.Stop(); var TimeToSearch = watch.ElapsedMilliseconds; Util.ValidateDataReturnResponseOk(a); //Now validate the return list var ResultCount = ((JArray)a.ObjectResponse["data"]["searchResults"]).Count; //assert it's not unbounded ResultCount.Should().BeLessOrEqualTo(1000); if (ResultCount > 999) { //assert the TotalResultsFound is greater than the results returned var TotalResultsFound = a.ObjectResponse["data"]["totalResultsFound"].Value(); //assert it's not unbounded TotalResultsFound.Should().BeGreaterThan(ResultCount); } //5456 ms is the longest I've seen in initial testing with all 1000 results so setting 10% above TimeToSearch.Should().BeLessThan(6000, "Constrained big data search should not be too slow"); }//eot [Fact] public async void UnboundBigDataSearchShouldBeRelativelyFast() { //THIS test is a bit different in that it relies partly on the big dataset for testing //so it has different paths depending upon if it's testing against the big data or not const string TEST_SEARCH_PHRASE = "et*"; //Now see if can find those objects with a phrase search dynamic SearchParameters = new JObject(); SearchParameters.phrase = TEST_SEARCH_PHRASE; SearchParameters.typeOnly = 0;//no type SearchParameters.maxResults = 0;//0=return all results var watch = new System.Diagnostics.Stopwatch(); watch.Start(); ApiResponse a = await Util.PostAsync("search", await Util.GetTokenAsync("superuser", "l3tm3in"), SearchParameters.ToString()); watch.Stop(); var TimeToSearch = watch.ElapsedMilliseconds; Util.ValidateDataReturnResponseOk(a); //Now validate the return list var ResultCount = ((JArray)a.ObjectResponse["data"]["searchResults"]).Count; //Set this time based on testing. 52385 Slowest run plus 10% TimeToSearch.Should().BeLessThan(57623, "Unconstrained big data search should not be too slow"); }//eot [Fact] public async void SearchForSerialFieldShouldWork() { //CREATE A WIDGET dynamic D = new JObject(); D.name = Util.Uniquify("Serial search test WIDGET"); D.customFields = Util.WidgetRequiredCustomFieldsJsonString(); D.dollarAmount = 1.11m; D.active = true; D.usertype = 1; D.notes = "Test for serial number search"; ApiResponse a = await Util.PostAsync("widget", await Util.GetTokenAsync("superuser", "l3tm3in"), D.ToString()); Util.ValidateDataReturnResponseOk(a); long MatchWidgetInSerialId = a.ObjectResponse["data"]["id"].Value(); var MatchWidgetSerial = a.ObjectResponse["data"]["serial"].Value(); //TODO: get this from the return object string SerialSearch = MatchWidgetSerial.ToString(); ; //Now see if can find those objects with a phrase search dynamic SearchParameters = new JObject(); SearchParameters.phrase = SerialSearch; SearchParameters.typeOnly = 0;//no type SearchParameters.maxResults = 0; a = await Util.PostAsync("search", await Util.GetTokenAsync("superuser", "l3tm3in"), SearchParameters.ToString()); Util.ValidateDataReturnResponseOk(a); //Now validate the return list ((JArray)a.ObjectResponse["data"]["searchResults"]).Count.Should().BeGreaterOrEqualTo(1); //Turn the list into an array of id's var v = ((JArray)a.ObjectResponse["data"]["searchResults"]); List MatchingIdList = new List(); foreach (JObject j in v) { MatchingIdList.Add(j["id"].Value()); } //Ensure the expected items are returned MatchingIdList.Should().Contain(MatchWidgetInSerialId, "ShouldContainMatchWidgetInSerialId"); }//eot [Fact] public async void CustomFieldSearchShouldWork() { /* Failing on huge data test after many hours: [xUnit.net 00:00:21.49] raven_integration.SearchOps.CustomFieldSearchShouldWork [FAIL] X raven_integration.SearchOps.CustomFieldSearchShouldWork [1s 84ms] Error Message: Expected MatchingIdList {109274L, 109095L, 108919L, 108740L, 108558L, 108382L, 108201L, 108021L, 107842L, 107663L, 107487L, 107305L, 107126L, 106947L, 106768L, 106589L, 106413L, 106233L, 106052L, 105873L, 105692L, 105510L, 105341L, 105157L, 104978L, 104799L, 104620L, 104441L, 104262L, 104084L, 103906L, 103725L, .468 more.} to contain 195378L because ShouldContainMatchingCustomFieldWidgetId. Stack Trace: at FluentAssertions.Execution.XUnit2TestFramework.Throw(String message) at FluentAssertions.Execution.TestFrameworkProvider.Throw(String message) at FluentAssertions.Execution.DefaultAssertionStrategy.HandleFailure(String message) at FluentAssertions.Execution.AssertionScope.FailWith(Func`1 failReasonFunc) at FluentAssertions.Collections.SelfReferencingCollectionAssertions`2.Contain(T expected, String because, Object[] becauseArgs) at raven_integration.SearchOps.CustomFieldSearchShouldWork() in c:\data\code\raven-test-integration\Search\SearchOps.cs:line 644 at System.Threading.Tasks.Task.<>c.b__139_0(Object state) Likely fix is to force order to most recent or simply to delete after test so they don't build up into a huge list */ //CREATE CUSTOM FIELD DATA TO SEARCH FOR (note must use already defined template so c2 is the text one) dynamic dCustomField = new JObject(); dCustomField.c1 = "2019-02-08T06:31:48.0019809Z"; dCustomField.c2 = "pelican beak nozzle"; dCustomField.c3 = "747"; dCustomField.c4 = "true"; dCustomField.c5 = "23.45"; //CREATE A WIDGET dynamic D = new JObject(); D.name = Util.Uniquify("CUSTOMFIELD search test WIDGET"); D.customFields = dCustomField.ToString(); D.dollarAmount = 1.11m; D.active = true; D.usertype = 1; D.notes = "This record will match in custom field"; ApiResponse a = await Util.PostAsync("widget", await Util.GetTokenAsync("superuser", "l3tm3in"), D.ToString()); Util.ValidateDataReturnResponseOk(a); long MatchingWidgetId = a.ObjectResponse["data"]["id"].Value(); //Now see if can find this object with a phrase search dynamic SearchParameters = new JObject(); SearchParameters.phrase = "beak pelican"; SearchParameters.typeOnly = 0;//no type a = await Util.PostAsync("search", await Util.GetTokenAsync("superuser", "l3tm3in"), SearchParameters.ToString()); Util.ValidateDataReturnResponseOk(a); //Now validate the return list ((JArray)a.ObjectResponse["data"]["searchResults"]).Count.Should().BeGreaterOrEqualTo(1);//might be a successive run and still have some left so account for more than one return //Turn the list into an array of id's var v = ((JArray)a.ObjectResponse["data"]["searchResults"]); List MatchingIdList = new List(); foreach (JObject j in v) { MatchingIdList.Add(j["id"].Value()); } //Ensure the expected items are returned MatchingIdList.Should().Contain(MatchingWidgetId, "ShouldContainMatchingCustomFieldWidgetId"); //Now delete the newly created one so the list doesn't build up and prevent this test from working in big runs... //DELETE ApiResponse DELETETestResponse = await Util.DeleteAsync("widget/" + MatchingWidgetId.ToString(), await Util.GetTokenAsync("superuser", "l3tm3in")); Util.ValidateHTTPStatusCode(DELETETestResponse, 204); }//eot [Fact] public async void DeletedObjectShouldRemoveKeywords() { string TEST_SEARCH_PHRASE = Util.Uniquify("qqwweerrttyy"); //CREATE A WIDGET dynamic d = new JObject(); d.name = Util.Uniquify("Wildcard endswith search test WIDGET"); d.customFields = Util.WidgetRequiredCustomFieldsJsonString(); d.dollarAmount = 1.11m; d.active = true; d.usertype = 1; d.notes = $"This record will match in notes: {TEST_SEARCH_PHRASE} there"; ApiResponse a = await Util.PostAsync("widget", await Util.GetTokenAsync("superuser", "l3tm3in"), d.ToString()); Util.ValidateDataReturnResponseOk(a); long WidgetId = (long)a.ObjectResponse["data"]["id"]; //Now see if can find with a phrase search dynamic SearchParameters = new JObject(); SearchParameters.phrase = TEST_SEARCH_PHRASE; SearchParameters.typeOnly = 0;//no type SearchParameters.maxResults = 0; a = await Util.PostAsync("search", await Util.GetTokenAsync("superuser", "l3tm3in"), SearchParameters.ToString()); Util.ValidateDataReturnResponseOk(a); //Now validate the return list (should be only one as it's super unique) ((JArray)a.ObjectResponse["data"]["searchResults"]).Count.Should().Be(1); //Now delete the widget and re-run the search a = await Util.DeleteAsync("widget/" + WidgetId.ToString(), await Util.GetTokenAsync("superuser", "l3tm3in")); Util.ValidateHTTPStatusCode(a, 204); a = await Util.PostAsync("search", await Util.GetTokenAsync("superuser", "l3tm3in"), SearchParameters.ToString()); Util.ValidateDataReturnResponseOk(a); //Now validate the return list (should be only one as it's super unique) ((JArray)a.ObjectResponse["data"]["searchResults"]).Count.Should().Be(0); }//eot //================================================== }//eoc }//eons