module SqliteTests open System.Text.Json open BitBadger.Documents open BitBadger.Documents.Sqlite open BitBadger.Documents.Tests open Expecto open Microsoft.Data.Sqlite open Types #nowarn "0044" (** UNIT TESTS **) /// Unit tests for the Query module of the SQLite library let queryTests = testList "Query" [ testList "whereByFields" [ test "succeeds for a single field when a logical operator is passed" { Expect.equal (Query.whereByFields Any [ { Field.GT "theField" 0 with ParameterName = Some "@test" } ]) "data->>'theField' > @test" "WHERE clause not correct" } test "succeeds for a single field when an existence operator is passed" { Expect.equal (Query.whereByFields Any [ Field.NEX "thatField" ]) "data->>'thatField' IS NULL" "WHERE clause not correct" } test "succeeds for a single field when a between operator is passed" { Expect.equal (Query.whereByFields All [ { Field.BT "aField" 50 99 with ParameterName = Some "@range" } ]) "data->>'aField' BETWEEN @rangemin AND @rangemax" "WHERE clause not correct" } test "succeeds for all multiple fields with logical operators" { Expect.equal (Query.whereByFields All [ Field.EQ "theFirst" "1"; Field.EQ "numberTwo" "2" ]) "data->>'theFirst' = @field0 AND data->>'numberTwo' = @field1" "WHERE clause not correct" } test "succeeds for any multiple fields with an existence operator" { Expect.equal (Query.whereByFields Any [ Field.NEX "thatField"; Field.GE "thisField" 18 ]) "data->>'thatField' IS NULL OR data->>'thisField' >= @field0" "WHERE clause not correct" } test "succeeds for all multiple fields with between operators" { Expect.equal (Query.whereByFields All [ Field.BT "aField" 50 99; Field.BT "anotherField" "a" "b" ]) "data->>'aField' BETWEEN @field0min AND @field0max AND data->>'anotherField' BETWEEN @field1min AND @field1max" "WHERE clause not correct" } ] test "whereById succeeds" { Expect.equal (Query.whereById "@id") "data->>'Id' = @id" "WHERE clause not correct" } test "patch succeeds" { Expect.equal (Query.patch SqliteDb.TableName) $"UPDATE {SqliteDb.TableName} SET data = json_patch(data, json(@data))" "Patch query not correct" } test "removeFields succeeds" { Expect.equal (Query.removeFields SqliteDb.TableName [ SqliteParameter("@a", "a"); SqliteParameter("@b", "b") ]) $"UPDATE {SqliteDb.TableName} SET data = json_remove(data, @a, @b)" "Field removal query not correct" } test "byId succeeds" { Expect.equal (Query.byId "test" "14") "test WHERE data->>'Id' = @id" "By-ID query not correct" } test "byFields succeeds" { Expect.equal (Query.byFields "unit" Any [ Field.GT "That" 14 ]) "unit WHERE data->>'That' > @field0" "By-Field query not correct" } test "Definition.ensureTable succeeds" { Expect.equal (Query.Definition.ensureTable "tbl") "CREATE TABLE IF NOT EXISTS tbl (data TEXT NOT NULL)" "CREATE TABLE statement not correct" } ] /// Unit tests for the Parameters module of the SQLite library let parametersTests = testList "Parameters" [ test "idParam succeeds" { let theParam = idParam 7 Expect.equal theParam.ParameterName "@id" "The parameter name is incorrect" Expect.equal theParam.Value "7" "The parameter value is incorrect" } test "jsonParam succeeds" { let theParam = jsonParam "@test" {| Nice = "job" |} Expect.equal theParam.ParameterName "@test" "The parameter name is incorrect" Expect.equal theParam.Value """{"Nice":"job"}""" "The parameter value is incorrect" } testList "addFieldParam" [ test "succeeds when adding a parameter" { let paramList = addFieldParam "@field" (Field.EQ "it" 99) [] Expect.hasLength paramList 1 "There should have been a parameter added" let theParam = Seq.head paramList Expect.equal theParam.ParameterName "@field" "The parameter name is incorrect" Expect.equal theParam.Value 99 "The parameter value is incorrect" } test "succeeds when not adding a parameter" { let paramList = addFieldParam "@it" (Field.NEX "Coffee") [] Expect.isEmpty paramList "There should not have been any parameters added" } ] test "noParams succeeds" { Expect.isEmpty noParams "The parameter list should have been empty" } ] // Results are exhaustively executed in the context of other tests (** INTEGRATION TESTS **) /// Load a table with the test documents let loadDocs () = backgroundTask { for doc in testDocuments do do! insert SqliteDb.TableName doc } /// Integration tests for the Configuration module of the SQLite library let configurationTests = testList "Configuration" [ test "useConnectionString / connectionString succeed" { try Configuration.useConnectionString "Data Source=test.db" Expect.equal Configuration.connectionString (Some "Data Source=test.db;Foreign Keys=True") "Connection string incorrect" finally Configuration.useConnectionString "Data Source=:memory:" } ] /// Integration tests for the Custom module of the SQLite library let customTests = testList "Custom" [ testList "single" [ testTask "succeeds when a row is found" { use! db = SqliteDb.BuildDb() do! loadDocs () let! doc = Custom.single $"SELECT data FROM {SqliteDb.TableName} WHERE data ->> 'Id' = @id" [ SqliteParameter("@id", "one") ] fromData Expect.isSome doc "There should have been a document returned" Expect.equal doc.Value.Id "one" "The incorrect document was returned" } testTask "succeeds when a row is not found" { use! db = SqliteDb.BuildDb() do! loadDocs () let! doc = Custom.single $"SELECT data FROM {SqliteDb.TableName} WHERE data ->> 'Id' = @id" [ SqliteParameter("@id", "eighty") ] fromData Expect.isNone doc "There should not have been a document returned" } ] testList "list" [ testTask "succeeds when data is found" { use! db = SqliteDb.BuildDb() do! loadDocs () let! docs = Custom.list (Query.find SqliteDb.TableName) [] fromData Expect.hasCountOf docs 5u (fun _ -> true) "There should have been 5 documents returned" } testTask "succeeds when data is not found" { use! db = SqliteDb.BuildDb() do! loadDocs () let! docs = Custom.list $"SELECT data FROM {SqliteDb.TableName} WHERE data ->> 'NumValue' > @value" [ SqliteParameter("@value", 100) ] fromData Expect.isEmpty docs "There should have been no documents returned" } ] testList "nonQuery" [ testTask "succeeds when operating on data" { use! db = SqliteDb.BuildDb() do! loadDocs () do! Custom.nonQuery $"DELETE FROM {SqliteDb.TableName}" [] let! remaining = Count.all SqliteDb.TableName Expect.equal remaining 0L "There should be no documents remaining in the table" } testTask "succeeds when no data matches where clause" { use! db = SqliteDb.BuildDb() do! loadDocs () do! Custom.nonQuery $"DELETE FROM {SqliteDb.TableName} WHERE data ->> 'NumValue' > @value" [ SqliteParameter("@value", 100) ] let! remaining = Count.all SqliteDb.TableName Expect.equal remaining 5L "There should be 5 documents remaining in the table" } ] testTask "scalar succeeds" { use! db = SqliteDb.BuildDb() let! nbr = Custom.scalar "SELECT 5 AS test_value" [] _.GetInt32(0) Expect.equal nbr 5 "The query should have returned the number 5" } ] /// Integration tests for the Definition module of the SQLite library let definitionTests = testList "Definition" [ testTask "ensureTable succeeds" { use! db = SqliteDb.BuildDb() let itExists (name: string) = Custom.scalar $"SELECT EXISTS (SELECT 1 FROM {SqliteDb.Catalog} WHERE name = @name) AS it" [ SqliteParameter("@name", name) ] toExists let! exists = itExists "ensured" let! alsoExists = itExists "idx_ensured_key" Expect.isFalse exists "The table should not exist already" Expect.isFalse alsoExists "The key index should not exist already" do! Definition.ensureTable "ensured" let! exists' = itExists "ensured" let! alsoExists' = itExists "idx_ensured_key" Expect.isTrue exists' "The table should now exist" Expect.isTrue alsoExists' "The key index should now exist" } testTask "ensureFieldIndex succeeds" { use! db = SqliteDb.BuildDb() let indexExists () = Custom.scalar $"SELECT EXISTS (SELECT 1 FROM {SqliteDb.Catalog} WHERE name = 'idx_ensured_test') AS it" [] toExists let! exists = indexExists () Expect.isFalse exists "The index should not exist already" do! Definition.ensureTable "ensured" do! Definition.ensureFieldIndex "ensured" "test" [ "Name"; "Age" ] let! exists' = indexExists () Expect.isTrue exists' "The index should now exist" } ] /// Integration tests for the (auto-opened) Document module of the SQLite library let documentTests = testList "Document" [ testList "insert" [ testTask "succeeds" { use! db = SqliteDb.BuildDb() let! before = Find.all SqliteDb.TableName Expect.equal before [] "There should be no documents in the table" let testDoc = { emptyDoc with Id = "turkey"; Sub = Some { Foo = "gobble"; Bar = "gobble" } } do! insert SqliteDb.TableName testDoc let! after = Find.all SqliteDb.TableName Expect.equal after [ testDoc ] "There should have been one document inserted" } testTask "fails for duplicate key" { use! db = SqliteDb.BuildDb() do! insert SqliteDb.TableName { emptyDoc with Id = "test" } Expect.throws (fun () -> insert SqliteDb.TableName {emptyDoc with Id = "test" } |> Async.AwaitTask |> Async.RunSynchronously) "An exception should have been raised for duplicate document ID insert" } testTask "succeeds when adding a numeric auto ID" { try Configuration.useAutoIdStrategy Number Configuration.useIdField "Key" use! db = SqliteDb.BuildDb() let! before = Count.all SqliteDb.TableName Expect.equal before 0L "There should be no documents in the table" do! insert SqliteDb.TableName { Key = 0; Text = "one" } do! insert SqliteDb.TableName { Key = 0; Text = "two" } do! insert SqliteDb.TableName { Key = 77; Text = "three" } do! insert SqliteDb.TableName { Key = 0; Text = "four" } let! after = Find.allOrdered SqliteDb.TableName [ Field.Named "Key" ] Expect.hasLength after 4 "There should have been 4 documents returned" Expect.equal (after |> List.map _.Key) [ 1; 2; 77; 78 ] "The IDs were not generated correctly" finally Configuration.useAutoIdStrategy Disabled Configuration.useIdField "Id" } testTask "succeeds when adding a GUID auto ID" { try Configuration.useAutoIdStrategy Guid use! db = SqliteDb.BuildDb() let! before = Count.all SqliteDb.TableName Expect.equal before 0L "There should be no documents in the table" do! insert SqliteDb.TableName { emptyDoc with Value = "one" } do! insert SqliteDb.TableName { emptyDoc with Value = "two" } do! insert SqliteDb.TableName { emptyDoc with Id = "abc123"; Value = "three" } do! insert SqliteDb.TableName { emptyDoc with Value = "four" } let! after = Find.all SqliteDb.TableName Expect.hasLength after 4 "There should have been 4 documents returned" Expect.hasCountOf after 3u (fun doc -> doc.Id.Length = 32) "Three of the IDs should have been GUIDs" Expect.hasCountOf after 1u (fun doc -> doc.Id = "abc123") "The provided ID should have been used as-is" finally Configuration.useAutoIdStrategy Disabled } testTask "succeeds when adding a RandomString auto ID" { try Configuration.useAutoIdStrategy RandomString Configuration.useIdStringLength 44 use! db = SqliteDb.BuildDb() let! before = Count.all SqliteDb.TableName Expect.equal before 0L "There should be no documents in the table" do! insert SqliteDb.TableName { emptyDoc with Value = "one" } do! insert SqliteDb.TableName { emptyDoc with Value = "two" } do! insert SqliteDb.TableName { emptyDoc with Id = "abc123"; Value = "three" } do! insert SqliteDb.TableName { emptyDoc with Value = "four" } let! after = Find.all SqliteDb.TableName Expect.hasLength after 4 "There should have been 4 documents returned" Expect.hasCountOf after 3u (fun doc -> doc.Id.Length = 44) "Three of the IDs should have been 44-character random strings" Expect.hasCountOf after 1u (fun doc -> doc.Id = "abc123") "The provided ID should have been used as-is" finally Configuration.useAutoIdStrategy Disabled Configuration.useIdStringLength 16 } ] testList "save" [ testTask "succeeds when a document is inserted" { use! db = SqliteDb.BuildDb() let! before = Find.all SqliteDb.TableName Expect.equal before [] "There should be no documents in the table" let testDoc = { emptyDoc with Id = "test"; Sub = Some { Foo = "a"; Bar = "b" } } do! save SqliteDb.TableName testDoc let! after = Find.all SqliteDb.TableName Expect.equal after [ testDoc ] "There should have been one document inserted" } testTask "succeeds when a document is updated" { use! db = SqliteDb.BuildDb() let testDoc = { emptyDoc with Id = "test"; Sub = Some { Foo = "a"; Bar = "b" } } do! insert SqliteDb.TableName testDoc let! before = Find.byId SqliteDb.TableName "test" Expect.isSome before "There should have been a document returned" Expect.equal before.Value testDoc "The document is not correct" let upd8Doc = { testDoc with Sub = Some { Foo = "c"; Bar = "d" } } do! save SqliteDb.TableName upd8Doc let! after = Find.byId SqliteDb.TableName "test" Expect.isSome after "There should have been a document returned post-update" Expect.equal after.Value upd8Doc "The updated document is not correct" } ] ] /// Integration tests for the Count module of the SQLite library let countTests = testList "Count" [ testTask "all succeeds" { use! db = SqliteDb.BuildDb() do! loadDocs () let! theCount = Count.all SqliteDb.TableName Expect.equal theCount 5L "There should have been 5 matching documents" } testList "byFields" [ testTask "succeeds for a numeric range" { use! db = SqliteDb.BuildDb() do! loadDocs () let! theCount = Count.byFields SqliteDb.TableName Any [ Field.BT "NumValue" 10 20 ] Expect.equal theCount 3L "There should have been 3 matching documents" } testTask "succeeds for a non-numeric range" { use! db = SqliteDb.BuildDb() do! loadDocs () let! theCount = Count.byFields SqliteDb.TableName Any [ Field.BT "Value" "aardvark" "apple" ] Expect.equal theCount 1L "There should have been 1 matching document" } ] ] /// Integration tests for the Exists module of the SQLite library let existsTests = testList "Exists" [ testList "byId" [ testTask "succeeds when a document exists" { use! db = SqliteDb.BuildDb() do! loadDocs () let! exists = Exists.byId SqliteDb.TableName "three" Expect.isTrue exists "There should have been an existing document" } testTask "succeeds when a document does not exist" { use! db = SqliteDb.BuildDb() do! loadDocs () let! exists = Exists.byId SqliteDb.TableName "seven" Expect.isFalse exists "There should not have been an existing document" } ] testList "byFields" [ testTask "succeeds when documents exist" { use! db = SqliteDb.BuildDb() do! loadDocs () let! exists = Exists.byFields SqliteDb.TableName Any [ Field.EQ "NumValue" 10 ] Expect.isTrue exists "There should have been existing documents" } testTask "succeeds when no matching documents exist" { use! db = SqliteDb.BuildDb() do! loadDocs () let! exists = Exists.byFields SqliteDb.TableName Any [ Field.LT "Nothing" "none" ] Expect.isFalse exists "There should not have been any existing documents" } ] ] /// Integration tests for the Find module of the SQLite library let findTests = testList "Find" [ testList "all" [ testTask "succeeds when there is data" { use! db = SqliteDb.BuildDb() do! insert SqliteDb.TableName { Foo = "one"; Bar = "two" } do! insert SqliteDb.TableName { Foo = "three"; Bar = "four" } do! insert SqliteDb.TableName { Foo = "five"; Bar = "six" } let! results = Find.all SqliteDb.TableName Expect.equal results [ { Foo = "one"; Bar = "two" }; { Foo = "three"; Bar = "four" }; { Foo = "five"; Bar = "six" } ] "There should have been 3 documents returned" } testTask "succeeds when there is no data" { use! db = SqliteDb.BuildDb() let! results = Find.all SqliteDb.TableName Expect.equal results [] "There should have been no documents returned" } ] testList "allOrdered" [ testTask "succeeds when ordering numerically" { use! db = SqliteDb.BuildDb() do! loadDocs () let! results = Find.allOrdered SqliteDb.TableName [ Field.Named "n:NumValue" ] Expect.hasLength results 5 "There should have been 5 documents returned" Expect.equal (results |> List.map _.Id |> String.concat "|") "one|three|two|four|five" "The documents were not ordered correctly" } testTask "succeeds when ordering numerically descending" { use! db = SqliteDb.BuildDb() do! loadDocs () let! results = Find.allOrdered SqliteDb.TableName [ Field.Named "n:NumValue DESC" ] Expect.hasLength results 5 "There should have been 5 documents returned" Expect.equal (results |> List.map _.Id |> String.concat "|") "five|four|two|three|one" "The documents were not ordered correctly" } testTask "succeeds when ordering alphabetically" { use! db = SqliteDb.BuildDb() do! loadDocs () let! results = Find.allOrdered SqliteDb.TableName [ Field.Named "Id DESC" ] Expect.hasLength results 5 "There should have been 5 documents returned" Expect.equal (results |> List.map _.Id |> String.concat "|") "two|three|one|four|five" "The documents were not ordered correctly" } ] testList "byId" [ testTask "succeeds when a document is found" { use! db = SqliteDb.BuildDb() do! loadDocs () let! doc = Find.byId SqliteDb.TableName "two" Expect.isSome doc "There should have been a document returned" Expect.equal doc.Value.Id "two" "The incorrect document was returned" } testTask "succeeds when a document is not found" { use! db = SqliteDb.BuildDb() do! loadDocs () let! doc = Find.byId SqliteDb.TableName "three hundred eighty-seven" Expect.isNone doc "There should not have been a document returned" } ] testList "byFields" [ testTask "succeeds when documents are found" { use! db = SqliteDb.BuildDb() do! loadDocs () let! docs = Find.byFields SqliteDb.TableName Any [ Field.GT "NumValue" 15 ] Expect.equal (List.length docs) 2 "There should have been two documents returned" } testTask "succeeds when documents are not found" { use! db = SqliteDb.BuildDb() do! loadDocs () let! docs = Find.byFields SqliteDb.TableName Any [ Field.GT "NumValue" 100 ] Expect.isTrue (List.isEmpty docs) "There should have been no documents returned" } ] testList "byFieldsOrdered" [ testTask "succeeds when sorting ascending" { use! db = SqliteDb.BuildDb() do! loadDocs () let! docs = Find.byFieldsOrdered SqliteDb.TableName Any [ Field.GT "NumValue" 15 ] [ Field.Named "Id" ] Expect.hasLength docs 2 "There should have been two documents returned" Expect.equal (docs |> List.map _.Id |> String.concat "|") "five|four" "The documents were not ordered correctly" } testTask "succeeds when sorting descending" { use! db = SqliteDb.BuildDb() do! loadDocs () let! docs = Find.byFieldsOrdered SqliteDb.TableName Any [ Field.GT "NumValue" 15 ] [ Field.Named "Id DESC" ] Expect.hasLength docs 2 "There should have been two documents returned" Expect.equal (docs |> List.map _.Id |> String.concat "|") "four|five" "The documents were not ordered correctly" } testTask "succeeds when sorting case-sensitively" { use! db = SqliteDb.BuildDb() do! loadDocs () let! docs = Find.byFieldsOrdered SqliteDb.TableName All [ Field.LE "NumValue" 10 ] [ Field.Named "Value" ] Expect.hasLength docs 3 "There should have been three documents returned" Expect.equal (docs |> List.map _.Id |> String.concat "|") "three|one|two" "Documents not ordered correctly" } testTask "succeeds when sorting case-insensitively" { use! db = SqliteDb.BuildDb() do! loadDocs () let! docs = Find.byFieldsOrdered SqliteDb.TableName All [ Field.LE "NumValue" 10 ] [ Field.Named "i:Value" ] Expect.hasLength docs 3 "There should have been three documents returned" Expect.equal (docs |> List.map _.Id |> String.concat "|") "three|two|one" "Documents not ordered correctly" } ] testList "firstByFields" [ testTask "succeeds when a document is found" { use! db = SqliteDb.BuildDb() do! loadDocs () let! doc = Find.firstByFields SqliteDb.TableName Any [ Field.EQ "Value" "another" ] Expect.isSome doc "There should have been a document returned" Expect.equal doc.Value.Id "two" "The incorrect document was returned" } testTask "succeeds when multiple documents are found" { use! db = SqliteDb.BuildDb() do! loadDocs () let! doc = Find.firstByFields SqliteDb.TableName Any [ Field.EQ "Sub.Foo" "green" ] Expect.isSome doc "There should have been a document returned" Expect.contains [ "two"; "four" ] doc.Value.Id "An incorrect document was returned" } testTask "succeeds when a document is not found" { use! db = SqliteDb.BuildDb() do! loadDocs () let! doc = Find.firstByFields SqliteDb.TableName Any [ Field.EQ "Value" "absent" ] Expect.isNone doc "There should not have been a document returned" } ] testList "firstByFieldsOrdered" [ testTask "succeeds when sorting ascending" { use! db = SqliteDb.BuildDb() do! loadDocs () let! doc = Find.firstByFieldsOrdered SqliteDb.TableName Any [ Field.EQ "Sub.Foo" "green" ] [ Field.Named "Sub.Bar" ] Expect.isSome doc "There should have been a document returned" Expect.equal "two" doc.Value.Id "An incorrect document was returned" } testTask "succeeds when sorting descending" { use! db = SqliteDb.BuildDb() do! loadDocs () let! doc = Find.firstByFieldsOrdered SqliteDb.TableName Any [ Field.EQ "Sub.Foo" "green" ] [ Field.Named "Sub.Bar DESC" ] Expect.isSome doc "There should have been a document returned" Expect.equal "four" doc.Value.Id "An incorrect document was returned" } ] ] /// Integration tests for the Update module of the SQLite library let updateTests = testList "Update" [ testList "byId" [ testTask "succeeds when a document is updated" { use! db = SqliteDb.BuildDb() do! loadDocs () let testDoc = { emptyDoc with Id = "one"; Sub = Some { Foo = "blue"; Bar = "red" } } do! Update.byId SqliteDb.TableName "one" testDoc let! after = Find.byId SqliteDb.TableName "one" Expect.isSome after "There should have been a document returned post-update" Expect.equal after.Value testDoc "The updated document is not correct" } testTask "succeeds when no document is updated" { use! db = SqliteDb.BuildDb() let! before = Find.all SqliteDb.TableName Expect.isEmpty before "There should have been no documents returned" // This not raising an exception is the test do! Update.byId SqliteDb.TableName "test" { emptyDoc with Id = "x"; Sub = Some { Foo = "blue"; Bar = "red" } } } ] testList "byFunc" [ testTask "succeeds when a document is updated" { use! db = SqliteDb.BuildDb() do! loadDocs () do! Update.byFunc SqliteDb.TableName (_.Id) { Id = "one"; Value = "le un"; NumValue = 1; Sub = None } let! after = Find.byId SqliteDb.TableName "one" Expect.isSome after "There should have been a document returned post-update" Expect.equal after.Value { Id = "one"; Value = "le un"; NumValue = 1; Sub = None } "The updated document is not correct" } testTask "succeeds when no document is updated" { use! db = SqliteDb.BuildDb() let! before = Find.all SqliteDb.TableName Expect.isEmpty before "There should have been no documents returned" // This not raising an exception is the test do! Update.byFunc SqliteDb.TableName (_.Id) { Id = "one"; Value = "le un"; NumValue = 1; Sub = None } } ] ] /// Integration tests for the Patch module of the SQLite library let patchTests = testList "Patch" [ testList "byId" [ testTask "succeeds when a document is updated" { use! db = SqliteDb.BuildDb() do! loadDocs () do! Patch.byId SqliteDb.TableName "one" {| NumValue = 44 |} let! after = Find.byId SqliteDb.TableName "one" Expect.isSome after "There should have been a document returned post-update" Expect.equal after.Value.NumValue 44 "The updated document is not correct" } testTask "succeeds when no document is updated" { use! db = SqliteDb.BuildDb() let! before = Find.all SqliteDb.TableName Expect.isEmpty before "There should have been no documents returned" // This not raising an exception is the test do! Patch.byId SqliteDb.TableName "test" {| Foo = "green" |} } ] testList "byFields" [ testTask "succeeds when a document is updated" { use! db = SqliteDb.BuildDb() do! loadDocs () do! Patch.byFields SqliteDb.TableName Any [ Field.EQ "Value" "purple" ] {| NumValue = 77 |} let! after = Count.byFields SqliteDb.TableName Any [ Field.EQ "NumValue" 77 ] Expect.equal after 2L "There should have been 2 documents returned" } testTask "succeeds when no document is updated" { use! db = SqliteDb.BuildDb() let! before = Find.all SqliteDb.TableName Expect.isEmpty before "There should have been no documents returned" // This not raising an exception is the test do! Patch.byFields SqliteDb.TableName Any [ Field.EQ "Value" "burgundy" ] {| Foo = "green" |} } ] ] /// Integration tests for the RemoveFields module of the SQLite library let removeFieldsTests = testList "RemoveFields" [ testList "byId" [ testTask "succeeds when fields is removed" { use! db = SqliteDb.BuildDb() do! loadDocs () do! RemoveFields.byId SqliteDb.TableName "two" [ "Sub"; "Value" ] try let! _ = Find.byId SqliteDb.TableName "two" Expect.isTrue false "The updated document should have failed to parse" with | :? JsonException -> () | exn as ex -> Expect.isTrue false $"Threw {ex.GetType().Name} ({ex.Message})" } testTask "succeeds when a field is not removed" { use! db = SqliteDb.BuildDb() do! loadDocs () // This not raising an exception is the test do! RemoveFields.byId SqliteDb.TableName "two" [ "AFieldThatIsNotThere" ] } testTask "succeeds when no document is matched" { use! db = SqliteDb.BuildDb() // This not raising an exception is the test do! RemoveFields.byId SqliteDb.TableName "two" [ "Value" ] } ] testList "byFields" [ testTask "succeeds when a field is removed" { use! db = SqliteDb.BuildDb() do! loadDocs () do! RemoveFields.byFields SqliteDb.TableName Any [ Field.EQ "NumValue" 17 ] [ "Sub" ] try let! _ = Find.byId SqliteDb.TableName "four" Expect.isTrue false "The updated document should have failed to parse" with | :? JsonException -> () | exn as ex -> Expect.isTrue false $"Threw {ex.GetType().Name} ({ex.Message})" } testTask "succeeds when a field is not removed" { use! db = SqliteDb.BuildDb() do! loadDocs () // This not raising an exception is the test do! RemoveFields.byFields SqliteDb.TableName Any [ Field.EQ "NumValue" 17 ] [ "Nothing" ] } testTask "succeeds when no document is matched" { use! db = SqliteDb.BuildDb() // This not raising an exception is the test do! RemoveFields.byFields SqliteDb.TableName Any [ Field.NE "Abracadabra" "apple" ] [ "Value" ] } ] ] /// Integration tests for the Delete module of the SQLite library let deleteTests = testList "Delete" [ testList "byId" [ testTask "succeeds when a document is deleted" { use! db = SqliteDb.BuildDb() do! loadDocs () do! Delete.byId SqliteDb.TableName "four" let! remaining = Count.all SqliteDb.TableName Expect.equal remaining 4L "There should have been 4 documents remaining" } testTask "succeeds when a document is not deleted" { use! db = SqliteDb.BuildDb() do! loadDocs () do! Delete.byId SqliteDb.TableName "thirty" let! remaining = Count.all SqliteDb.TableName Expect.equal remaining 5L "There should have been 5 documents remaining" } ] testList "byFields" [ testTask "succeeds when documents are deleted" { use! db = SqliteDb.BuildDb() do! loadDocs () do! Delete.byFields SqliteDb.TableName Any [ Field.NE "Value" "purple" ] let! remaining = Count.all SqliteDb.TableName Expect.equal remaining 2L "There should have been 2 documents remaining" } testTask "succeeds when documents are not deleted" { use! db = SqliteDb.BuildDb() do! loadDocs () do! Delete.byFields SqliteDb.TableName Any [ Field.EQ "Value" "crimson" ] let! remaining = Count.all SqliteDb.TableName Expect.equal remaining 5L "There should have been 5 documents remaining" } ] ] /// All tests for the SQLite library let all = testList "Sqlite" [ testList "Unit" [ queryTests; parametersTests ] testSequenced <| testList "Integration" [ configurationTests customTests definitionTests documentTests countTests existsTests findTests updateTests patchTests removeFieldsTests deleteTests test "clean up database" { Configuration.useConnectionString "data source=:memory:" } ] ]