using System; using Xunit; using Newtonsoft.Json.Linq; using FluentAssertions; namespace raven_integration { public class UserCrud { /// /// Test all CRUD routes for a User /// [Fact] public async void CRUD() { //CREATE dynamic D1 = new JObject(); D1.name = Util.Uniquify("First Test User"); D1.active = true; D1.login = Util.Uniquify("LOGIN"); D1.password = Util.Uniquify("PASSWORD"); D1.roles = 0;//norole D1.localeId = 1;//random locale D1.userType = 3;//non scheduleable //Required by form custom rules D1.notes = "notes"; D1.customFields = Util.UserRequiredCustomFieldsJsonString(); ApiResponse R1 = await Util.PostAsync("User", await Util.GetTokenAsync("manager", "l3tm3in"), D1.ToString()); Util.ValidateDataReturnResponseOk(R1); long d1Id = R1.ObjectResponse["data"]["id"].Value(); dynamic D2 = new JObject(); D2.name = Util.Uniquify("Second Test User"); //Required by form custom rules D2.notes = "notes"; D2.customFields = Util.UserRequiredCustomFieldsJsonString(); D2.active = true; D2.login = Util.Uniquify("LOGIN"); D2.password = Util.Uniquify("PASSWORD"); D2.roles = 0;//norole D2.localeId = 1;//random locale D2.userType = 3;//non scheduleable ApiResponse R2 = await Util.PostAsync("User", await Util.GetTokenAsync("manager", "l3tm3in"), D2.ToString()); Util.ValidateDataReturnResponseOk(R2); long d2Id = R2.ObjectResponse["data"]["id"].Value(); //RETRIEVE //Get one ApiResponse R3 = await Util.GetAsync("User/" + d2Id.ToString(), await Util.GetTokenAsync("manager", "l3tm3in")); Util.ValidateDataReturnResponseOk(R3); R3.ObjectResponse["data"]["name"].Value().Should().Be(D2.name.ToString()); //UPDATE //PUT //update w2id D2.name = Util.Uniquify("UPDATED VIA PUT SECOND TEST User"); D2.concurrencyToken = R2.ObjectResponse["data"]["concurrencyToken"].Value(); ApiResponse PUTTestResponse = await Util.PutAsync("User/" + d2Id.ToString(), await Util.GetTokenAsync("manager", "l3tm3in"), D2.ToString()); Util.ValidateHTTPStatusCode(PUTTestResponse, 200); //check PUT worked ApiResponse checkPUTWorked = await Util.GetAsync("User/" + d2Id.ToString(), await Util.GetTokenAsync("manager", "l3tm3in")); Util.ValidateNoErrorInResponse(checkPUTWorked); checkPUTWorked.ObjectResponse["data"]["name"].Value().Should().Be(D2.name.ToString()); uint concurrencyToken = PUTTestResponse.ObjectResponse["data"]["concurrencyToken"].Value(); //PATCH var newName = Util.Uniquify("UPDATED VIA PATCH SECOND TEST User"); string patchJson = "[{\"value\": \"" + newName + "\",\"path\": \"/name\",\"op\": \"replace\"}]"; ApiResponse PATCHTestResponse = await Util.PatchAsync("User/" + d2Id.ToString() + "/" + concurrencyToken.ToString(), await Util.GetTokenAsync("manager", "l3tm3in"), patchJson); Util.ValidateHTTPStatusCode(PATCHTestResponse, 200); //check PATCH worked ApiResponse checkPATCHWorked = await Util.GetAsync("User/" + d2Id.ToString(), await Util.GetTokenAsync("manager", "l3tm3in")); Util.ValidateNoErrorInResponse(checkPATCHWorked); checkPATCHWorked.ObjectResponse["data"]["name"].Value().Should().Be(newName); //DELETE ApiResponse DELETETestResponse = await Util.DeleteAsync("User/" + d2Id.ToString(), await Util.GetTokenAsync("manager", "l3tm3in")); Util.ValidateHTTPStatusCode(DELETETestResponse, 204); } /// /// /// [Fact] public async void UserWithActivityShouldNotBeDeleteable() { ApiResponse a = await Util.DeleteAsync("User/1", await Util.GetTokenAsync("manager", "l3tm3in")); Util.ValidateErrorCodeResponse(a, 2200, 400); a.ObjectResponse["error"]["details"][0]["message"].Value().Should().Contain("LT:ErrorDBForeignKeyViolation"); } /// /// Test not found /// [Fact] public async void GetNonExistentItemShouldError() { //Get non existant //Should return status code 404, api error code 2010 ApiResponse R = await Util.GetAsync("User/999999", await Util.GetTokenAsync("manager", "l3tm3in")); Util.ValidateResponseNotFound(R); } /// /// Test bad modelstate /// [Fact] public async void GetBadModelStateShouldError() { //Get non existant //Should return status code 400, api error code 2200 and a first target in details of "id" ApiResponse R = await Util.GetAsync("User/2q2", await Util.GetTokenAsync("manager", "l3tm3in")); Util.ValidateBadModelStateResponse(R, "id"); } /// /// /// [Fact] public async void PutConcurrencyViolationShouldFail() { //CREATE dynamic D = new JObject(); D.name = Util.Uniquify("PutConcurrencyViolationShouldFail"); D.notes = "notes"; D.customFields = Util.UserRequiredCustomFieldsJsonString(); D.active = true; D.login = Util.Uniquify("LOGIN"); D.password = Util.Uniquify("PASSWORD"); D.roles = 0;//norole D.localeId = 1;//random locale D.userType = 3;//non scheduleable ApiResponse R = await Util.PostAsync("User", await Util.GetTokenAsync("manager", "l3tm3in"), D.ToString()); Util.ValidateDataReturnResponseOk(R); long D1Id = R.ObjectResponse["data"]["id"].Value(); uint OriginalConcurrencyToken = R.ObjectResponse["data"]["concurrencyToken"].Value(); //UPDATE //PUT D.name = Util.Uniquify("PutConcurrencyViolationShouldFail UPDATE VIA PUT "); D.concurrencyToken = OriginalConcurrencyToken - 1;//bad token ApiResponse PUTTestResponse = await Util.PutAsync("User/" + D1Id.ToString(), await Util.GetTokenAsync("manager", "l3tm3in"), D.ToString()); Util.ValidateConcurrencyError(PUTTestResponse); } /// /// /// [Fact] public async void PatchConcurrencyViolationShouldFail() { //CREATE dynamic D = new JObject(); D.name = Util.Uniquify("PatchConcurrencyViolationShouldFail"); D.active = true; D.login = Util.Uniquify("LOGIN"); D.password = Util.Uniquify("PASSWORD"); D.roles = 0;//norole D.localeId = 1;//random locale D.userType = 3;//non scheduleable //Required by form custom rules D.notes = "notes"; D.customFields = Util.UserRequiredCustomFieldsJsonString(); ApiResponse R = await Util.PostAsync("User", await Util.GetTokenAsync("manager", "l3tm3in"), D.ToString()); Util.ValidateDataReturnResponseOk(R); long w2Id = R.ObjectResponse["data"]["id"].Value(); uint OriginalConcurrencyToken = R.ObjectResponse["data"]["concurrencyToken"].Value(); //PATCH var newName = Util.Uniquify("PutConcurrencyViolationShouldFail UPDATED VIA PATCH"); string patchJson = "[{\"value\": \"" + newName + "\",\"path\": \"/name\",\"op\": \"replace\"}]"; ApiResponse PATCHTestResponse = await Util.PatchAsync("User/" + w2Id.ToString() + "/" + (OriginalConcurrencyToken - 1).ToString(), await Util.GetTokenAsync("manager", "l3tm3in"), patchJson); Util.ValidateConcurrencyError(PATCHTestResponse); } /// /// /// [Fact] public async void DisallowedPatchAttemptsShouldFail() { //CREATE dynamic D = new JObject(); D.name = Util.Uniquify("DisallowedPatchAttemptsShouldFail"); D.active = true; D.login = Util.Uniquify("LOGIN"); D.password = Util.Uniquify("PASSWORD"); D.roles = 0;//norole D.localeId = 1;//random locale D.userType = 3;//non scheduleable //Required by form custom rules D.notes = "notes"; D.customFields = Util.UserRequiredCustomFieldsJsonString(); ApiResponse R = await Util.PostAsync("User", await Util.GetTokenAsync("manager", "l3tm3in"), D.ToString()); Util.ValidateDataReturnResponseOk(R); long w2Id = R.ObjectResponse["data"]["id"].Value(); uint OriginalConcurrencyToken = R.ObjectResponse["data"]["concurrencyToken"].Value(); //PATCH attempt on Id string patchJson = "[{\"value\": \"0\",\"path\": \"/id\",\"op\": \"replace\"}]"; ApiResponse PATCHTestResponse = await Util.PatchAsync("User/" + w2Id.ToString() + "/" + (OriginalConcurrencyToken - 1).ToString(), await Util.GetTokenAsync("manager", "l3tm3in"), patchJson); Util.ValidateErrorCodeResponse(PATCHTestResponse, 2200, 400); //PATCH attempt add field patchJson = "[{\"value\": \"0\",\"path\": \"/bogus\",\"op\": \"add\"}]"; PATCHTestResponse = await Util.PatchAsync("User/" + w2Id.ToString() + "/" + (OriginalConcurrencyToken - 1).ToString(), await Util.GetTokenAsync("manager", "l3tm3in"), patchJson); Util.ValidateErrorCodeResponse(PATCHTestResponse, 2200, 400); //PATCH attempt remove name field patchJson = "[{\"path\": \"/name\",\"op\": \"remove\"}]"; PATCHTestResponse = await Util.PatchAsync("User/" + w2Id.ToString() + "/" + (OriginalConcurrencyToken - 1).ToString(), await Util.GetTokenAsync("manager", "l3tm3in"), patchJson); Util.ValidateErrorCodeResponse(PATCHTestResponse, 2200, 400); } /// /// /// [Fact] public async void PatchPasswordShouldWork() { //CREATE dynamic D = new JObject(); D.name = Util.Uniquify("PatchPasswordShouldWork"); D.active = true; D.login = Util.Uniquify("LOGIN"); D.password = Util.Uniquify("PASSWORD"); D.roles = 0;//norole D.localeId = 1;//random locale D.userType = 3;//non scheduleable //Required by form custom rules D.notes = "notes"; D.customFields = Util.UserRequiredCustomFieldsJsonString(); ApiResponse R = await Util.PostAsync("User", await Util.GetTokenAsync("manager", "l3tm3in"), D.ToString()); Util.ValidateDataReturnResponseOk(R); long UserId = R.ObjectResponse["data"]["id"].Value(); uint OriginalConcurrencyToken = R.ObjectResponse["data"]["concurrencyToken"].Value(); //Test can login dynamic DCreds = new JObject(); DCreds.password = D.password; DCreds.login = D.login; R = await Util.PostAsync("Auth", null, DCreds.ToString()); Util.ValidateDataReturnResponseOk(R); //PUT var NewPassword = "NEW_PASSWORD"; D.password = NewPassword; D.concurrencyToken = OriginalConcurrencyToken; R = await Util.PutAsync("User/" + UserId.ToString(), await Util.GetTokenAsync("manager", "l3tm3in"), D.ToString()); Util.ValidateDataReturnResponseOk(R); //Test can login with new creds //dynamic DCreds = new JObject(); DCreds.password = NewPassword; DCreds.login = D.login; R = await Util.PostAsync("Auth", null, DCreds.ToString()); Util.ValidateDataReturnResponseOk(R); } /// /// /// [Fact] public async void PutPasswordShouldWork() { //CREATE dynamic D = new JObject(); D.name = Util.Uniquify("PutPasswordShouldWork"); D.active = true; D.login = Util.Uniquify("LOGIN"); D.password = Util.Uniquify("PASSWORD"); D.roles = 0;//norole D.localeId = 1;//random locale D.userType = 3;//non scheduleable //Required by form custom rules D.notes = "notes"; D.customFields = Util.UserRequiredCustomFieldsJsonString(); ApiResponse R = await Util.PostAsync("User", await Util.GetTokenAsync("manager", "l3tm3in"), D.ToString()); Util.ValidateDataReturnResponseOk(R); long UserId = R.ObjectResponse["data"]["id"].Value(); uint OriginalConcurrencyToken = R.ObjectResponse["data"]["concurrencyToken"].Value(); //Test can login dynamic DCreds = new JObject(); DCreds.password = D.password; DCreds.login = D.login; R = await Util.PostAsync("Auth", null, DCreds.ToString()); Util.ValidateDataReturnResponseOk(R); //PATCH var newPassword = "NEW_PASSWORD"; string patchJson = "[{\"value\": \"" + newPassword + "\",\"path\": \"/password\",\"op\": \"replace\"}]"; R = await Util.PatchAsync("User/" + UserId.ToString() + "/" + OriginalConcurrencyToken.ToString(), await Util.GetTokenAsync("manager", "l3tm3in"), patchJson); Util.ValidateDataReturnResponseOk(R); //Test can login with new creds //dynamic DCreds = new JObject(); DCreds.password = newPassword; DCreds.login = D.login; R = await Util.PostAsync("Auth", null, DCreds.ToString()); Util.ValidateDataReturnResponseOk(R); } /// /// /// [Fact] public async void UserListFilterAndSortWorks() { var ObjectNameStart = Util.Uniquify("UserListFilterAndSortWorks"); //CREATE 3 TEST OBJECTS TO TEST ORDER long FirstInOrdertId = 0; long SecondInOrderId = 0; long ThirdInOrderId = 0; dynamic d = new JObject(); d.name = Util.Uniquify(ObjectNameStart); d.active = false; d.login = Util.Uniquify("LOGIN"); d.password = Util.Uniquify("PASSWORD"); d.roles = 0;//norole d.localeId = 1;//random locale d.userType = 3;//non scheduleable //Required by form custom rules d.notes = "notes"; d.customFields = Util.UserRequiredCustomFieldsJsonString(); ApiResponse a = await Util.PostAsync("User", await Util.GetTokenAsync("manager", "l3tm3in"), d.ToString()); Util.ValidateDataReturnResponseOk(a); FirstInOrdertId = a.ObjectResponse["data"]["id"].Value(); d = new JObject(); d.name = Util.Uniquify(ObjectNameStart); d.login = Util.Uniquify("LOGIN"); d.password = Util.Uniquify("PASSWORD"); d.roles = 0;//norole d.localeId = 1;//random locale d.userType = 3;//non scheduleable d.active = true; //Required by form custom rules d.notes = "notes"; d.customFields = Util.UserRequiredCustomFieldsJsonString(); a = await Util.PostAsync("User", await Util.GetTokenAsync("manager", "l3tm3in"), d.ToString()); Util.ValidateDataReturnResponseOk(a); ThirdInOrderId = a.ObjectResponse["data"]["id"].Value(); d = new JObject(); d.name = Util.Uniquify(ObjectNameStart); d.login = Util.Uniquify("LOGIN"); d.password = Util.Uniquify("PASSWORD"); d.roles = 0;//norole d.localeId = 1;//random locale d.userType = 3;//non scheduleable d.active = false; //Required by form custom rules d.notes = "notes"; d.customFields = Util.UserRequiredCustomFieldsJsonString(); a = await Util.PostAsync("User", await Util.GetTokenAsync("manager", "l3tm3in"), d.ToString()); Util.ValidateDataReturnResponseOk(a); SecondInOrderId = a.ObjectResponse["data"]["id"].Value(); //CREATE FILTER d = new JObject(); d.name = Util.Uniquify(ObjectNameStart); d["public"] = true; d.listKey = "UserDataList"; //FILTER IN BY NAME FOR TESTING THIS RUN ONLY dynamic dfilter = new JArray(); //name starts with filter to constrict to widgets that this test block created only dynamic DataFilterNameStart = new JObject(); DataFilterNameStart.fld = "username"; DataFilterNameStart.op = Util.OpStartsWith; DataFilterNameStart.value = ObjectNameStart; dfilter.Add(DataFilterNameStart); d.filter = dfilter.ToString(); dynamic dsortarray = new JArray(); //SORT ORDER ################### //sort by active then by ID dynamic dsort = new JObject(); dsort.fld = "useractive"; dsort.dir = "+"; dsortarray.Add(dsort); dsort = new JObject(); dsort.fld = "df"; dsort.dir = "+"; dsortarray.Add(dsort); d.sort = dsortarray.ToString(); a = await Util.PostAsync("DataListFilter", await Util.GetTokenAsync("manager", "l3tm3in"), d.ToString()); Util.ValidateDataReturnResponseOk(a); long DataFilterId = a.ObjectResponse["data"]["id"].Value(); //NOW FETCH WIDGET LIST WITH FILTER UserDataList a = await Util.GetAsync($"DataList/list?DataListKey=UserDataList&Offset=0&Limit=999&DataFilterId={DataFilterId.ToString()}", await Util.GetTokenAsync("manager", "l3tm3in")); Util.ValidateDataReturnResponseOk(a); Util.ValidateHTTPStatusCode(a, 200); //assert contains exactly 3 records ((JArray)a.ObjectResponse["data"]).Count.Should().Be(3); //assert the order returned a.ObjectResponse["data"][0][0]["v"].Value().Should().Be(FirstInOrdertId); a.ObjectResponse["data"][1][0]["v"].Value().Should().Be(SecondInOrderId); a.ObjectResponse["data"][2][0]["v"].Value().Should().Be(ThirdInOrderId); a = await Util.DeleteAsync("User/" + FirstInOrdertId.ToString(), await Util.GetTokenAsync("manager", "l3tm3in")); Util.ValidateHTTPStatusCode(a, 204); a = await Util.DeleteAsync("User/" + SecondInOrderId.ToString(), await Util.GetTokenAsync("manager", "l3tm3in")); Util.ValidateHTTPStatusCode(a, 204); a = await Util.DeleteAsync("User/" + ThirdInOrderId.ToString(), await Util.GetTokenAsync("manager", "l3tm3in")); Util.ValidateHTTPStatusCode(a, 204); //DELETE DATAFILTER a = await Util.DeleteAsync("DataListFilter/" + DataFilterId.ToString(), await Util.GetTokenAsync("manager", "l3tm3in")); Util.ValidateHTTPStatusCode(a, 204); } //DEPRECATED THESE BECAUSE THEY DON'T TEST ANYTHING UNIQUELY THAT ISN'T ALREADY COVERED BY THE DATALISTFILTER AND ORDER BY TESTS ELSEWHERE //AND THE PICKLIST IS NOW MINI FORMAT OF REGULAR LIST // /// // /// // /// // [Fact] // public async void UserPickListDefaultSortNoFilterWorks() // { // var RouteName = "User";//########## // //NOW FETCH LIST WITH FILTER // ApiResponse a = await Util.GetAsync($"{RouteName}/picklist?Offset=0&Limit=999", await Util.GetTokenAsync("manager", "l3tm3in")); // Util.ValidateDataReturnResponseOk(a); // Util.ValidateHTTPStatusCode(a, 200); // //assert contains at least 3 records // var ItemCount = ((JArray)a.ObjectResponse["data"]).Count; // ItemCount.Should().BeGreaterOrEqualTo(3); // var firstName = a.ObjectResponse["data"][0]["name"].Value().Replace(" ", ""); // var secondName = a.ObjectResponse["data"][1]["name"].Value().Replace(" ", ""); // int comparison = String.Compare(firstName, secondName, comparisonType: StringComparison.OrdinalIgnoreCase); // comparison.Should().BeNegative(); // } // /// // /// // /// // [Fact] // public async void UserPickListSortByFieldAscendingWorks() // { // var NameStart = Util.Uniquify("UserPickListSortByFieldAscendingWorks"); // //CREATE 3 TEST objects TO TEST ORDER // long FirstInOrderId = 0; // long SecondInOrderId = 0; // long ThirdInOrderId = 0; // dynamic d = new JObject(); // d.name = Util.Uniquify(NameStart); // d.active = false; // d.login = Util.Uniquify("LOGIN"); // d.password = Util.Uniquify("PASSWORD"); // d.roles = 0;//norole // d.localeId = 1;//random locale // d.userType = 3;//non scheduleable // //Required by form custom rules // d.notes = "notes"; // d.customFields = Util.UserRequiredCustomFieldsJsonString(); // ApiResponse a = await Util.PostAsync("User", await Util.GetTokenAsync("manager", "l3tm3in"), d.ToString()); // Util.ValidateDataReturnResponseOk(a); // ThirdInOrderId = a.ObjectResponse["data"]["id"].Value(); // d = new JObject(); // d.name = Util.Uniquify(NameStart); // d.active = true; // d.login = Util.Uniquify("LOGIN"); // d.password = Util.Uniquify("PASSWORD"); // d.roles = 0;//norole // d.localeId = 1;//random locale // d.userType = 2;//non scheduleable // //Required by form custom rules // d.notes = "notes"; // d.customFields = Util.UserRequiredCustomFieldsJsonString(); // a = await Util.PostAsync("User", await Util.GetTokenAsync("manager", "l3tm3in"), d.ToString()); // Util.ValidateDataReturnResponseOk(a); // SecondInOrderId = a.ObjectResponse["data"]["id"].Value(); // d = new JObject(); // d.name = Util.Uniquify(NameStart); // d.active = false; // d.login = Util.Uniquify("LOGIN"); // d.password = Util.Uniquify("PASSWORD"); // d.roles = 0;//norole // d.localeId = 1;//random locale // d.userType = 1;//non scheduleable // //Required by form custom rules // d.notes = "notes"; // d.customFields = Util.UserRequiredCustomFieldsJsonString(); // a = await Util.PostAsync("User", await Util.GetTokenAsync("manager", "l3tm3in"), d.ToString()); // Util.ValidateDataReturnResponseOk(a); // FirstInOrderId = a.ObjectResponse["data"]["id"].Value(); // //CREATE FILTER // d = new JObject(); // d.name = Util.Uniquify(NameStart); // d["public"] = true; // d.listKey = "user"; // //FILTER IN BY NAME FOR TESTING THIS RUN ONLY // dynamic dfilter = new JArray(); // //name starts with filter to constrict to widgets that this test block created only // dynamic DataFilterNameStart = new JObject(); // DataFilterNameStart.fld = "name"; // DataFilterNameStart.op = Util.OpStartsWith; // DataFilterNameStart.value = NameStart; // dfilter.Add(DataFilterNameStart); // d.filter = dfilter.ToString(); // //SORT ORDER ################### // dynamic dsortarray = new JArray(); // dynamic dsort = new JObject(); // dsort.fld = "usertype"; // dsort.dir = "+"; // dsortarray.Add(dsort); // d.sort = dsortarray.ToString(); // a = await Util.PostAsync("DataListFilter", await Util.GetTokenAsync("manager", "l3tm3in"), d.ToString()); // Util.ValidateDataReturnResponseOk(a); // long DataFilterId = a.ObjectResponse["data"]["id"].Value(); // //NOW FETCH WIDGET LIST WITH FILTER // a = await Util.GetAsync($"User/picklist?Offset=0&Limit=999&DataFilterId={DataFilterId.ToString()}", await Util.GetTokenAsync("manager", "l3tm3in")); // Util.ValidateDataReturnResponseOk(a); // Util.ValidateHTTPStatusCode(a, 200); // //assert contains exactly 3 records // ((JArray)a.ObjectResponse["data"]).Count.Should().Be(3); // //assert the order returned // a.ObjectResponse["data"][0]["id"].Value().Should().Be(FirstInOrderId); // a.ObjectResponse["data"][1]["id"].Value().Should().Be(SecondInOrderId); // a.ObjectResponse["data"][2]["id"].Value().Should().Be(ThirdInOrderId); // a = await Util.DeleteAsync("User/" + FirstInOrderId.ToString(), await Util.GetTokenAsync("manager", "l3tm3in")); // Util.ValidateHTTPStatusCode(a, 204); // a = await Util.DeleteAsync("User/" + SecondInOrderId.ToString(), await Util.GetTokenAsync("manager", "l3tm3in")); // Util.ValidateHTTPStatusCode(a, 204); // a = await Util.DeleteAsync("User/" + ThirdInOrderId.ToString(), await Util.GetTokenAsync("manager", "l3tm3in")); // Util.ValidateHTTPStatusCode(a, 204); // //DELETE DATAFILTER // a = await Util.DeleteAsync("DataListFilter/" + DataFilterId.ToString(), await Util.GetTokenAsync("manager", "l3tm3in")); // Util.ValidateHTTPStatusCode(a, 204); // } //================================================== }//eoc }//eons